288 lines
12 KiB
TypeScript
288 lines
12 KiB
TypeScript
import React from 'react';
|
|
import CardBox from '../CardBox';
|
|
import ImageField from '../ImageField';
|
|
import dataFormatter from '../../helpers/dataFormatter';
|
|
import {saveFile} from "../../helpers/fileSaver";
|
|
import ListActionsPopover from "../ListActionsPopover";
|
|
import {useAppSelector} from "../../stores/hooks";
|
|
import {Pagination} from "../Pagination";
|
|
import LoadingSpinner from "../LoadingSpinner";
|
|
import Link from 'next/link';
|
|
|
|
import {hasPermission} from "../../helpers/userPermissions";
|
|
|
|
|
|
type Props = {
|
|
trades: any[];
|
|
loading: boolean;
|
|
onDelete: (id: string) => void;
|
|
currentPage: number;
|
|
numPages: number;
|
|
onPageChange: (page: number) => void;
|
|
};
|
|
|
|
const ListTrades = ({ trades, loading, onDelete, currentPage, numPages, onPageChange }: Props) => {
|
|
|
|
const currentUser = useAppSelector((state) => state.auth.currentUser);
|
|
const hasUpdatePermission = hasPermission(currentUser, 'UPDATE_TRADES')
|
|
|
|
const corners = useAppSelector((state) => state.style.corners);
|
|
const bgColor = useAppSelector((state) => state.style.cardsColor);
|
|
|
|
|
|
return (
|
|
<>
|
|
<div className='relative overflow-x-auto p-4 space-y-4'>
|
|
{loading && <LoadingSpinner />}
|
|
{!loading && trades.map((item) => (
|
|
<div key={item.id}>
|
|
<CardBox hasTable isList className={'rounded shadow-none'}>
|
|
<div className={`flex ${bgColor} ${corners !== 'rounded-full' ? corners : 'rounded-3xl'} dark:bg-dark-900 border border-gray-600 items-center overflow-hidden`}>
|
|
|
|
<Link
|
|
href={`/trades/trades-view/?id=${item.id}`}
|
|
className={
|
|
'flex-1 px-4 py-6 h-24 flex divide-x-2 divide-gray-600 items-center overflow-hidden`}> dark:divide-dark-700 overflow-x-auto'
|
|
}
|
|
>
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>TradingAccount</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.trading_accountsOneListFormatter(item.trading_account) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Instrument</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.instrumentsOneListFormatter(item.instrument) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Setup</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.setupsOneListFormatter(item.setup) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Direction</p>
|
|
<p className={'line-clamp-2'}>{ item.direction }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Status</p>
|
|
<p className={'line-clamp-2'}>{ item.status }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Timeframe</p>
|
|
<p className={'line-clamp-2'}>{ item.timeframe }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>EntryAt</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.dateTimeFormatter(item.entry_at) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>ExitAt</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.dateTimeFormatter(item.exit_at) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>EntryPrice</p>
|
|
<p className={'line-clamp-2'}>{ item.entry_price }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>ExitPrice</p>
|
|
<p className={'line-clamp-2'}>{ item.exit_price }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>StopLossPrice</p>
|
|
<p className={'line-clamp-2'}>{ item.stop_loss_price }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>TakeProfitPrice</p>
|
|
<p className={'line-clamp-2'}>{ item.take_profit_price }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>PositionSize</p>
|
|
<p className={'line-clamp-2'}>{ item.position_size }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>PositionSizeUnit</p>
|
|
<p className={'line-clamp-2'}>{ item.position_size_unit }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Fees</p>
|
|
<p className={'line-clamp-2'}>{ item.fees }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Slippage</p>
|
|
<p className={'line-clamp-2'}>{ item.slippage }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>PnLAmount</p>
|
|
<p className={'line-clamp-2'}>{ item.pnl_amount }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>PnLPercent</p>
|
|
<p className={'line-clamp-2'}>{ item.pnl_percent }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>RiskAmount</p>
|
|
<p className={'line-clamp-2'}>{ item.risk_amount }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>RiskRewardRatio</p>
|
|
<p className={'line-clamp-2'}>{ item.rr_ratio }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>MaxAdverseExcursion</p>
|
|
<p className={'line-clamp-2'}>{ item.mae_amount }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>MaxFavorableExcursion</p>
|
|
<p className={'line-clamp-2'}>{ item.mfe_amount }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>DurationMinutes</p>
|
|
<p className={'line-clamp-2'}>{ item.duration_minutes }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>IsWin</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.booleanFormatter(item.is_win) }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Notes</p>
|
|
<p className={'line-clamp-2'}>{ item.notes }</p>
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div className={'flex-1 px-3'}>
|
|
<p className={'text-xs text-gray-500 '}>Tags</p>
|
|
<p className={'line-clamp-2'}>{ dataFormatter.tagsManyListFormatter(item.tags).join(', ')}</p>
|
|
</div>
|
|
|
|
|
|
|
|
</Link>
|
|
<ListActionsPopover
|
|
onDelete={onDelete}
|
|
itemId={item.id}
|
|
pathEdit={`/trades/trades-edit/?id=${item.id}`}
|
|
pathView={`/trades/trades-view/?id=${item.id}`}
|
|
|
|
hasUpdatePermission={hasUpdatePermission}
|
|
|
|
/>
|
|
</div>
|
|
</CardBox>
|
|
</div>
|
|
))}
|
|
{!loading && trades.length === 0 && (
|
|
<div className='col-span-full flex items-center justify-center h-40'>
|
|
<p className=''>No data to display</p>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className={'flex items-center justify-center my-6'}>
|
|
<Pagination
|
|
currentPage={currentPage}
|
|
numPages={numPages}
|
|
setCurrentPage={onPageChange}
|
|
/>
|
|
</div>
|
|
</>
|
|
)
|
|
};
|
|
|
|
export default ListTrades |