2023-06-27 00:10:22 +00:00
|
|
|
import { useDataGridEvents } from '@vegaprotocol/datagrid';
|
2023-08-02 15:34:04 +00:00
|
|
|
import { Filter, OrderListManager } from '@vegaprotocol/orders';
|
2023-06-27 00:10:22 +00:00
|
|
|
import { Splash } from '@vegaprotocol/ui-toolkit';
|
|
|
|
import { useVegaWallet } from '@vegaprotocol/wallet';
|
2023-10-16 18:18:26 +00:00
|
|
|
import { useNavigateWithMeta } from '../../lib/hooks/use-market-click-handler';
|
2023-06-27 00:10:22 +00:00
|
|
|
import { create } from 'zustand';
|
|
|
|
import { persist } from 'zustand/middleware';
|
|
|
|
import type { DataGridStore } from '../../stores/datagrid-store-slice';
|
|
|
|
import { OrderStatus } from '@vegaprotocol/types';
|
2023-10-16 18:18:26 +00:00
|
|
|
import { Links } from '../../lib/links';
|
2023-11-16 03:10:39 +00:00
|
|
|
import { useT } from '../../lib/use-t';
|
2023-06-27 00:10:22 +00:00
|
|
|
|
2023-11-16 03:10:39 +00:00
|
|
|
const resolveNoRowsMessage = (
|
|
|
|
filter: Filter | undefined,
|
|
|
|
t: ReturnType<typeof useT>
|
|
|
|
) => {
|
2023-10-19 13:45:55 +00:00
|
|
|
switch (filter) {
|
|
|
|
case Filter.Open:
|
|
|
|
return t('No open orders');
|
|
|
|
case Filter.Closed:
|
|
|
|
return t('No closed orders');
|
|
|
|
case Filter.Rejected:
|
|
|
|
return t('No rejected orders');
|
|
|
|
default:
|
|
|
|
return t('No orders');
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-06-27 00:10:22 +00:00
|
|
|
export const FilterStatusValue = {
|
|
|
|
[Filter.Open]: [OrderStatus.STATUS_ACTIVE, OrderStatus.STATUS_PARKED],
|
|
|
|
[Filter.Closed]: [
|
|
|
|
OrderStatus.STATUS_CANCELLED,
|
|
|
|
OrderStatus.STATUS_EXPIRED,
|
|
|
|
OrderStatus.STATUS_FILLED,
|
|
|
|
OrderStatus.STATUS_PARTIALLY_FILLED,
|
|
|
|
OrderStatus.STATUS_STOPPED,
|
|
|
|
],
|
|
|
|
[Filter.Rejected]: [OrderStatus.STATUS_REJECTED],
|
|
|
|
};
|
|
|
|
|
|
|
|
export interface OrderContainerProps {
|
|
|
|
filter?: Filter;
|
|
|
|
}
|
|
|
|
|
2023-09-28 19:14:32 +00:00
|
|
|
const AUTO_SIZE_COLUMNS = ['instrument-code'];
|
|
|
|
|
2023-08-10 15:02:46 +00:00
|
|
|
export const OrdersContainer = ({ filter }: OrderContainerProps) => {
|
2023-11-16 03:10:39 +00:00
|
|
|
const t = useT();
|
2023-06-27 00:10:22 +00:00
|
|
|
const { pubKey, isReadOnly } = useVegaWallet();
|
2023-10-16 18:18:26 +00:00
|
|
|
const navigate = useNavigateWithMeta();
|
2023-06-27 00:10:22 +00:00
|
|
|
const { gridState, updateGridState } = useOrderListGridState(filter);
|
2023-09-28 19:14:32 +00:00
|
|
|
const gridStoreCallbacks = useDataGridEvents(
|
|
|
|
gridState,
|
|
|
|
(newState) => {
|
|
|
|
updateGridState(filter, newState);
|
|
|
|
},
|
|
|
|
AUTO_SIZE_COLUMNS
|
|
|
|
);
|
2023-06-27 00:10:22 +00:00
|
|
|
|
|
|
|
if (!pubKey) {
|
|
|
|
return <Splash>{t('Please connect Vega wallet')}</Splash>;
|
|
|
|
}
|
2023-11-16 03:10:39 +00:00
|
|
|
const noRowsMessage = resolveNoRowsMessage(filter, t);
|
2023-06-27 00:10:22 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<OrderListManager
|
|
|
|
partyId={pubKey}
|
|
|
|
filter={filter}
|
2023-10-16 18:18:26 +00:00
|
|
|
onMarketClick={(marketId, metaKey) => {
|
|
|
|
navigate(Links.MARKET(marketId), metaKey);
|
|
|
|
}}
|
|
|
|
onOrderTypeClick={(marketId, metaKey) =>
|
|
|
|
navigate(Links.LIQUIDITY(marketId), metaKey)
|
|
|
|
}
|
2023-06-27 00:10:22 +00:00
|
|
|
isReadOnly={isReadOnly}
|
|
|
|
gridProps={gridStoreCallbacks}
|
2023-10-19 13:45:55 +00:00
|
|
|
noRowsMessage={noRowsMessage}
|
2023-06-27 00:10:22 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const STORAGE_KEY = 'vega_order_list_store';
|
|
|
|
const useOrderListStore = create<{
|
|
|
|
open: DataGridStore;
|
|
|
|
closed: DataGridStore;
|
|
|
|
rejected: DataGridStore;
|
|
|
|
all: DataGridStore;
|
|
|
|
update: (filter: Filter | undefined, gridStore: DataGridStore) => void;
|
|
|
|
}>()(
|
|
|
|
persist(
|
|
|
|
(set) => ({
|
|
|
|
open: {},
|
|
|
|
closed: {},
|
|
|
|
rejected: {},
|
|
|
|
all: {},
|
|
|
|
update: (filter, newStore) => {
|
|
|
|
switch (filter) {
|
|
|
|
case Filter.Open: {
|
|
|
|
set((curr) => ({
|
|
|
|
open: {
|
|
|
|
...curr.open,
|
|
|
|
...newStore,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case Filter.Closed: {
|
|
|
|
set((curr) => ({
|
|
|
|
closed: {
|
|
|
|
...curr.closed,
|
|
|
|
...newStore,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case Filter.Rejected: {
|
|
|
|
set((curr) => ({
|
|
|
|
rejected: {
|
|
|
|
...curr.rejected,
|
|
|
|
...newStore,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
case undefined: {
|
|
|
|
set((curr) => ({
|
|
|
|
all: {
|
|
|
|
...curr.all,
|
|
|
|
...newStore,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
{
|
|
|
|
name: STORAGE_KEY,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
export const useOrderListGridState = (filter: Filter | undefined) => {
|
|
|
|
const updateGridState = useOrderListStore((store) => store.update);
|
|
|
|
const gridState = useOrderListStore((store) => {
|
|
|
|
// Return the column/filter state for the given filter but ensuring that
|
|
|
|
// each filter controlled by the tab is always applied
|
|
|
|
switch (filter) {
|
|
|
|
case Filter.Open: {
|
|
|
|
return {
|
|
|
|
columnState: store.open.columnState,
|
|
|
|
filterModel: {
|
|
|
|
...store.open.filterModel,
|
|
|
|
status: {
|
|
|
|
value: FilterStatusValue[Filter.Open],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
case Filter.Closed: {
|
|
|
|
return {
|
|
|
|
columnState: store.closed.columnState,
|
|
|
|
filterModel: {
|
|
|
|
...store.closed.filterModel,
|
|
|
|
status: {
|
|
|
|
value: FilterStatusValue[Filter.Closed],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
case Filter.Rejected: {
|
|
|
|
return {
|
|
|
|
columnState: store.rejected.columnState,
|
|
|
|
filterModel: {
|
|
|
|
...store.rejected.filterModel,
|
|
|
|
status: {
|
|
|
|
value: FilterStatusValue[Filter.Rejected],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
return store.all;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return { gridState, updateGridState };
|
|
|
|
};
|