@open-mercato/core 0.4.6-develop-6247c21d52 → 0.4.6-develop-4cfe4a0f36

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.
@@ -70,7 +70,7 @@ function CurrenciesPage() {
70
70
  const handleSetBase = React.useCallback(
71
71
  async (row) => {
72
72
  try {
73
- const call = await apiCall("/api/currencies", {
73
+ const call = await apiCall("/api/currencies/currencies", {
74
74
  method: "PUT",
75
75
  headers: { "Content-Type": "application/json" },
76
76
  body: JSON.stringify({ id: row.id, isBase: true })
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/currencies/backend/currencies/page.tsx"],
4
- "sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { BooleanIcon } from '@open-mercato/ui/backend/ValueIcons'\nimport { Plus, Star } from 'lucide-react'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\n\ntype CurrencyRow = {\n id: string\n code: string\n name: string\n symbol: string | null\n decimalPlaces: number\n isBase: boolean\n isActive: boolean\n organizationId: string\n tenantId: string\n createdAt: string\n updatedAt: string\n}\n\ntype ResponsePayload = {\n items: CurrencyRow[]\n total: number\n page: number\n totalPages: number\n}\n\nexport default function CurrenciesPage() {\n const t = useT()\n const { confirm: confirmDialog, ConfirmDialogElement } = useConfirmDialog()\n const [rows, setRows] = React.useState<CurrencyRow[]>([])\n const [page, setPage] = React.useState(1)\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filters, setFilters] = React.useState<FilterValues>({})\n const [isLoading, setIsLoading] = React.useState(true)\n const [reloadToken, setReloadToken] = React.useState(0)\n const scopeVersion = useOrganizationScopeVersion()\n\n React.useEffect(() => {\n let cancelled = false\n async function load() {\n setIsLoading(true)\n try {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', '50')\n if (search) params.set('search', search)\n if (filters.isBase === true) params.set('isBase', 'true')\n if (filters.isActive === 'true') params.set('isActive', 'true')\n if (filters.isActive === 'false') params.set('isActive', 'false')\n\n const fallback: ResponsePayload = { items: [], total: 0, page, totalPages: 1 }\n const call = await apiCall<ResponsePayload>(\n `/api/currencies/currencies?${params.toString()}`,\n undefined,\n { fallback }\n )\n\n if (!call.ok) {\n flash(t('currencies.list.error.load'), 'error')\n return\n }\n\n const payload = call.result ?? fallback\n if (!cancelled) {\n setRows(Array.isArray(payload.items) ? payload.items : [])\n setTotal(payload.total || 0)\n setTotalPages(payload.totalPages || 1)\n }\n } catch (error) {\n if (!cancelled) {\n flash(t('currencies.list.error.load'), 'error')\n }\n } finally {\n if (!cancelled) setIsLoading(false)\n }\n }\n load()\n return () => {\n cancelled = true\n }\n }, [page, search, filters, reloadToken, scopeVersion, t])\n\n const handleSetBase = React.useCallback(\n async (row: CurrencyRow) => {\n try {\n const call = await apiCall('/api/currencies', {\n method: 'PUT',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ id: row.id, isBase: true }),\n })\n\n if (!call.ok) {\n flash(t('currencies.flash.baseSetError'), 'error')\n return\n }\n\n flash(t('currencies.flash.baseSet'), 'success')\n setReloadToken((token) => token + 1)\n } catch (error) {\n flash(t('currencies.flash.baseSetError'), 'error')\n }\n },\n [t]\n )\n\n const handleDelete = React.useCallback(\n async (row: CurrencyRow) => {\n const confirmed = await confirmDialog({\n title: t('currencies.list.confirmDelete', { code: row.code }),\n variant: 'destructive',\n })\n if (!confirmed) return\n\n try {\n const call = await apiCall(`/api/currencies/currencies`, {\n method: 'DELETE',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ id: row.id, organizationId: row.organizationId, tenantId: row.tenantId }),\n })\n\n if (!call.ok) {\n flash(t('currencies.flash.deleteError'), 'error')\n return\n }\n\n flash(t('currencies.flash.deleted'), 'success')\n setReloadToken((token) => token + 1)\n } catch (error) {\n flash(t('currencies.flash.deleteError'), 'error')\n }\n },\n [t, confirmDialog]\n )\n\n const columns = React.useMemo<ColumnDef<CurrencyRow>[]>(\n () => [\n {\n accessorKey: 'code',\n header: t('currencies.list.columns.code'),\n cell: ({ row }) => (\n <div className=\"flex items-center gap-2\">\n <span className=\"font-mono font-medium\">{row.original.code}</span>\n {row.original.isBase && (\n <Badge variant=\"default\" className=\"gap-1\">\n <Star className=\"h-3 w-3\" />\n {t('currencies.list.base')}\n </Badge>\n )}\n </div>\n ),\n },\n {\n accessorKey: 'name',\n header: t('currencies.list.columns.name'),\n },\n {\n accessorKey: 'symbol',\n header: t('currencies.list.columns.symbol'),\n cell: ({ row }) => row.original.symbol || '\u2014',\n },\n {\n accessorKey: 'decimalPlaces',\n header: t('currencies.list.columns.decimalPlaces'),\n },\n {\n accessorKey: 'isActive',\n header: t('currencies.list.columns.active'),\n enableSorting: false,\n cell: ({ getValue }) => <BooleanIcon value={Boolean(getValue())} />,\n },\n {\n accessorKey: 'createdAt',\n header: t('currencies.list.columns.createdAt'),\n cell: ({ row }) => {\n const date = row.original.createdAt\n return date ? new Date(date).toLocaleString() : '\u2014'\n },\n },\n {\n accessorKey: 'updatedAt',\n header: t('currencies.list.columns.updatedAt'),\n cell: ({ row }) => {\n const date = row.original.updatedAt\n return date ? new Date(date).toLocaleString() : '\u2014'\n },\n },\n ],\n [t]\n )\n\n const filterDefs = React.useMemo<FilterDef[]>(\n () => [\n {\n id: 'isBase',\n label: t('currencies.list.filters.baseOnly'),\n type: 'checkbox',\n },\n {\n id: 'isActive',\n label: t('currencies.list.filters.status'),\n type: 'select',\n options: [\n { label: t('currencies.list.filters.all'), value: '' },\n { label: t('currencies.list.filters.active'), value: 'true' },\n { label: t('currencies.list.filters.inactive'), value: 'false' },\n ],\n },\n ],\n [t]\n )\n\n return (\n <Page>\n <PageBody>\n <DataTable\n title={t('currencies.list.title')}\n columns={columns}\n data={rows}\n searchValue={search}\n onSearchChange={(value) => {\n setSearch(value)\n setPage(1)\n }}\n searchPlaceholder={t('currencies.list.searchPlaceholder')}\n filters={filterDefs}\n filterValues={filters}\n onFiltersApply={(values) => {\n setFilters(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilters({})\n setPage(1)\n }}\n actions={\n <Button asChild>\n <Link href=\"/backend/currencies/create\">\n <Plus className=\"mr-2 h-4 w-4\" />\n {t('currencies.list.actions.create')}\n </Link>\n </Button>\n }\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'edit',\n label: t('common.edit'),\n href: `/backend/currencies/${row.id}`,\n },\n ...(!row.isBase\n ? [\n {\n id: 'set-base',\n label: t('currencies.list.actions.setBase'),\n onSelect: () => handleSetBase(row),\n },\n ]\n : []),\n ...(!row.isBase\n ? [\n {\n id: 'delete',\n label: t('common.delete'),\n destructive: true,\n onSelect: () => handleDelete(row),\n },\n ]\n : []),\n ]}\n />\n )}\n pagination={{ page, pageSize: 50, total, totalPages, onPageChange: setPage }}\n isLoading={isLoading}\n perspective={{ tableId: 'currencies.list' }}\n />\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["'use client'\n\nimport * as React from 'react'\nimport Link from 'next/link'\nimport { Page, PageBody } from '@open-mercato/ui/backend/Page'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport type { ColumnDef } from '@tanstack/react-table'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { BooleanIcon } from '@open-mercato/ui/backend/ValueIcons'\nimport { Plus, Star } from 'lucide-react'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useOrganizationScopeVersion } from '@open-mercato/shared/lib/frontend/useOrganizationScope'\nimport { useConfirmDialog } from '@open-mercato/ui/backend/confirm-dialog'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\n\ntype CurrencyRow = {\n id: string\n code: string\n name: string\n symbol: string | null\n decimalPlaces: number\n isBase: boolean\n isActive: boolean\n organizationId: string\n tenantId: string\n createdAt: string\n updatedAt: string\n}\n\ntype ResponsePayload = {\n items: CurrencyRow[]\n total: number\n page: number\n totalPages: number\n}\n\nexport default function CurrenciesPage() {\n const t = useT()\n const { confirm: confirmDialog, ConfirmDialogElement } = useConfirmDialog()\n const [rows, setRows] = React.useState<CurrencyRow[]>([])\n const [page, setPage] = React.useState(1)\n const [total, setTotal] = React.useState(0)\n const [totalPages, setTotalPages] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filters, setFilters] = React.useState<FilterValues>({})\n const [isLoading, setIsLoading] = React.useState(true)\n const [reloadToken, setReloadToken] = React.useState(0)\n const scopeVersion = useOrganizationScopeVersion()\n\n React.useEffect(() => {\n let cancelled = false\n async function load() {\n setIsLoading(true)\n try {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', '50')\n if (search) params.set('search', search)\n if (filters.isBase === true) params.set('isBase', 'true')\n if (filters.isActive === 'true') params.set('isActive', 'true')\n if (filters.isActive === 'false') params.set('isActive', 'false')\n\n const fallback: ResponsePayload = { items: [], total: 0, page, totalPages: 1 }\n const call = await apiCall<ResponsePayload>(\n `/api/currencies/currencies?${params.toString()}`,\n undefined,\n { fallback }\n )\n\n if (!call.ok) {\n flash(t('currencies.list.error.load'), 'error')\n return\n }\n\n const payload = call.result ?? fallback\n if (!cancelled) {\n setRows(Array.isArray(payload.items) ? payload.items : [])\n setTotal(payload.total || 0)\n setTotalPages(payload.totalPages || 1)\n }\n } catch (error) {\n if (!cancelled) {\n flash(t('currencies.list.error.load'), 'error')\n }\n } finally {\n if (!cancelled) setIsLoading(false)\n }\n }\n load()\n return () => {\n cancelled = true\n }\n }, [page, search, filters, reloadToken, scopeVersion, t])\n\n const handleSetBase = React.useCallback(\n async (row: CurrencyRow) => {\n try {\n const call = await apiCall('/api/currencies/currencies', {\n method: 'PUT',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ id: row.id, isBase: true }),\n })\n\n if (!call.ok) {\n flash(t('currencies.flash.baseSetError'), 'error')\n return\n }\n\n flash(t('currencies.flash.baseSet'), 'success')\n setReloadToken((token) => token + 1)\n } catch (error) {\n flash(t('currencies.flash.baseSetError'), 'error')\n }\n },\n [t]\n )\n\n const handleDelete = React.useCallback(\n async (row: CurrencyRow) => {\n const confirmed = await confirmDialog({\n title: t('currencies.list.confirmDelete', { code: row.code }),\n variant: 'destructive',\n })\n if (!confirmed) return\n\n try {\n const call = await apiCall(`/api/currencies/currencies`, {\n method: 'DELETE',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ id: row.id, organizationId: row.organizationId, tenantId: row.tenantId }),\n })\n\n if (!call.ok) {\n flash(t('currencies.flash.deleteError'), 'error')\n return\n }\n\n flash(t('currencies.flash.deleted'), 'success')\n setReloadToken((token) => token + 1)\n } catch (error) {\n flash(t('currencies.flash.deleteError'), 'error')\n }\n },\n [t, confirmDialog]\n )\n\n const columns = React.useMemo<ColumnDef<CurrencyRow>[]>(\n () => [\n {\n accessorKey: 'code',\n header: t('currencies.list.columns.code'),\n cell: ({ row }) => (\n <div className=\"flex items-center gap-2\">\n <span className=\"font-mono font-medium\">{row.original.code}</span>\n {row.original.isBase && (\n <Badge variant=\"default\" className=\"gap-1\">\n <Star className=\"h-3 w-3\" />\n {t('currencies.list.base')}\n </Badge>\n )}\n </div>\n ),\n },\n {\n accessorKey: 'name',\n header: t('currencies.list.columns.name'),\n },\n {\n accessorKey: 'symbol',\n header: t('currencies.list.columns.symbol'),\n cell: ({ row }) => row.original.symbol || '\u2014',\n },\n {\n accessorKey: 'decimalPlaces',\n header: t('currencies.list.columns.decimalPlaces'),\n },\n {\n accessorKey: 'isActive',\n header: t('currencies.list.columns.active'),\n enableSorting: false,\n cell: ({ getValue }) => <BooleanIcon value={Boolean(getValue())} />,\n },\n {\n accessorKey: 'createdAt',\n header: t('currencies.list.columns.createdAt'),\n cell: ({ row }) => {\n const date = row.original.createdAt\n return date ? new Date(date).toLocaleString() : '\u2014'\n },\n },\n {\n accessorKey: 'updatedAt',\n header: t('currencies.list.columns.updatedAt'),\n cell: ({ row }) => {\n const date = row.original.updatedAt\n return date ? new Date(date).toLocaleString() : '\u2014'\n },\n },\n ],\n [t]\n )\n\n const filterDefs = React.useMemo<FilterDef[]>(\n () => [\n {\n id: 'isBase',\n label: t('currencies.list.filters.baseOnly'),\n type: 'checkbox',\n },\n {\n id: 'isActive',\n label: t('currencies.list.filters.status'),\n type: 'select',\n options: [\n { label: t('currencies.list.filters.all'), value: '' },\n { label: t('currencies.list.filters.active'), value: 'true' },\n { label: t('currencies.list.filters.inactive'), value: 'false' },\n ],\n },\n ],\n [t]\n )\n\n return (\n <Page>\n <PageBody>\n <DataTable\n title={t('currencies.list.title')}\n columns={columns}\n data={rows}\n searchValue={search}\n onSearchChange={(value) => {\n setSearch(value)\n setPage(1)\n }}\n searchPlaceholder={t('currencies.list.searchPlaceholder')}\n filters={filterDefs}\n filterValues={filters}\n onFiltersApply={(values) => {\n setFilters(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilters({})\n setPage(1)\n }}\n actions={\n <Button asChild>\n <Link href=\"/backend/currencies/create\">\n <Plus className=\"mr-2 h-4 w-4\" />\n {t('currencies.list.actions.create')}\n </Link>\n </Button>\n }\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'edit',\n label: t('common.edit'),\n href: `/backend/currencies/${row.id}`,\n },\n ...(!row.isBase\n ? [\n {\n id: 'set-base',\n label: t('currencies.list.actions.setBase'),\n onSelect: () => handleSetBase(row),\n },\n ]\n : []),\n ...(!row.isBase\n ? [\n {\n id: 'delete',\n label: t('common.delete'),\n destructive: true,\n onSelect: () => handleDelete(row),\n },\n ]\n : []),\n ]}\n />\n )}\n pagination={{ page, pageSize: 50, total, totalPages, onPageChange: setPage }}\n isLoading={isLoading}\n perspective={{ tableId: 'currencies.list' }}\n />\n </PageBody>\n {ConfirmDialogElement}\n </Page>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@open-mercato/core",
3
- "version": "0.4.6-develop-6247c21d52",
3
+ "version": "0.4.6-develop-4cfe4a0f36",
4
4
  "type": "module",
5
5
  "main": "./dist/index.js",
6
6
  "scripts": {
@@ -207,7 +207,7 @@
207
207
  }
208
208
  },
209
209
  "dependencies": {
210
- "@open-mercato/shared": "0.4.6-develop-6247c21d52",
210
+ "@open-mercato/shared": "0.4.6-develop-4cfe4a0f36",
211
211
  "@types/html-to-text": "^9.0.4",
212
212
  "@types/semver": "^7.5.8",
213
213
  "@xyflow/react": "^12.6.0",
@@ -99,7 +99,7 @@ export default function CurrenciesPage() {
99
99
  const handleSetBase = React.useCallback(
100
100
  async (row: CurrencyRow) => {
101
101
  try {
102
- const call = await apiCall('/api/currencies', {
102
+ const call = await apiCall('/api/currencies/currencies', {
103
103
  method: 'PUT',
104
104
  headers: { 'Content-Type': 'application/json' },
105
105
  body: JSON.stringify({ id: row.id, isBase: true }),