mars-v2-frontend/components/Borrow/BorrowTable.tsx
Gustavo Mauricio bbbdca6950
MP-1227: Borrow Page (#24)
* added icon for atom and tokenInfo data update

* borrow page initial commit

* feat: borrow funds to ca and wallet

* close borrow module on tx success

* feat: repay funds initial setup

* repay funds action hook

* repay slider. module state on borrow page component

* styling: minor tweak to text colors

* limit manual input on repay to max value

* borrow funds component slider initial

* style: max button typography

* AssetRow extracted to separate file. organize imports

* ContainerSecondary component added

* loading indicator for pending actions

* style: progress bar colors

* tanstack table added

* tanstack react-table dependency missing

* table cleanup and layout adjustments

* fix account stats formula and update market data to match spreadsheet

* calculate max borrow amount hook

* reset borrow and repay components on account change

* max borrow amount decimals. memorized return

* hook tanstack data with real data

* redefine borrowedAssetsMap to map

* update max borrow amount formulas

* remove unnecessary table component. refactor borrow table
2022-10-20 16:39:21 +01:00

198 lines
5.2 KiB
TypeScript

import React from 'react'
import Image from 'next/image'
import {
ColumnDef,
flexRender,
getCoreRowModel,
getSortedRowModel,
SortingState,
useReactTable,
} from '@tanstack/react-table'
import { ChevronDownIcon, ChevronUpIcon } from '@heroicons/react/24/solid'
import { formatCurrency } from 'utils/formatters'
import AssetRow from './AssetRow'
interface Market {
denom: string
symbol: string
icon: string
chain: string
borrowed: {
amount: number
value: number
} | null
borrowRate: number
marketLiquidity: number
}
// const data = [
// {
// denom: 'uosmo',
// symbol: 'OSMO',
// icon: '/tokens/osmo.svg',
// chain: 'Osmosis',
// borrowed: {
// amount: 2.005494,
// value: 2.2060434000000004,
// },
// borrowRate: 0.1,
// marketLiquidity: 1000000,
// },
// {
// denom: 'ibc/27394FB092D2ECCD56123C74F36E4C1F926001CEADA9CA97EA622B25F41E5EB2',
// symbol: 'ATOM',
// icon: '/tokens/atom.svg',
// chain: 'Cosmos',
// borrowed: null,
// borrowRate: 0.25,
// marketLiquidity: 1000,
// },
// {
// denom: 'uusdc',
// symbol: 'USDC',
// icon: '/tokens/atom.svg',
// chain: 'Ethereum',
// borrowed: {
// amount: 100,
// value: 99.9776,
// },
// borrowRate: 0.35,
// marketLiquidity: 333,
// },
// ]
type Props = {
data: Market[]
onBorrowClick: (denom: string) => void
onRepayClick: (denom: string, repayAmount: number) => void
}
const BorrowTable = ({ data, onBorrowClick, onRepayClick }: Props) => {
const [sorting, setSorting] = React.useState<SortingState>([])
const columns = React.useMemo<ColumnDef<Market>[]>(
() => [
{
header: 'Asset',
id: 'symbol',
accessorFn: (row) => (
<div className="flex flex-1 items-center">
<Image src={row.icon} alt="token" width={32} height={32} />
<div className="pl-2">
<div>{row.symbol}</div>
<div className="text-xs">{row.chain}</div>
</div>
</div>
),
cell: (info) => info.getValue(),
},
{
accessorKey: 'borrowRate',
header: 'Borrow Rate',
accessorFn: (row) => (
<div className="flex flex-1 items-center text-xs">
{row.borrowRate ? `${(row.borrowRate * 100).toFixed(2)}%` : '-'}
</div>
),
cell: (info) => info.getValue(),
},
{
accessorKey: 'age',
header: 'Borrowed',
accessorFn: (row) => (
<div className="flex flex-1 items-center text-xs">
{row.borrowed ? (
<div>
<div className="font-bold">{row.borrowed.amount}</div>
<div>{formatCurrency(row.borrowed.value)}</div>
</div>
) : (
'-'
)}
</div>
),
cell: (info) => info.getValue(),
},
{
accessorKey: 'marketLiquidity',
header: 'Liquidity Available',
},
{
accessorKey: 'status',
enableSorting: false,
header: 'Manage',
width: 150,
cell: ({ row }) => (
<div className="flex items-center justify-end">
{row.getIsExpanded() ? (
<ChevronUpIcon className="w-5" />
) : (
<ChevronDownIcon className="w-5" />
)}
</div>
),
},
],
[]
)
const table = useReactTable({
data,
columns,
state: {
sorting,
},
onSortingChange: setSorting,
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
debugTable: true,
})
return (
<div className="w-full table-fixed border-spacing-10 text-sm">
{table.getHeaderGroups().map((headerGroup) => (
<div
key={headerGroup.id}
className="mb-2 flex rounded-md bg-[#D8DAEA] px-4 py-2 text-xs text-[#585A74]/50"
>
{headerGroup.headers.map((header) => {
return (
<div key={header.id} className={`${header.index === 4 ? 'w-[50px]' : 'flex-1'}`}>
{header.isPlaceholder ? null : (
<div
{...{
className: header.column.getCanSort() ? 'cursor-pointer select-none' : '',
onClick: header.column.getToggleSortingHandler(),
}}
>
{flexRender(header.column.columnDef.header, header.getContext())}
{{
asc: ' 🔼',
desc: ' 🔽',
}[header.column.getIsSorted() as string] ?? null}
</div>
)}
</div>
)
})}
</div>
))}
<div className="flex flex-col gap-2">
{table.getRowModel().rows.map((row) => {
return (
<AssetRow
key={row.index}
data={row.original}
onBorrowClick={() => onBorrowClick(row.original.denom)}
onRepayClick={(repayAmount: number) => onRepayClick(row.original.denom, repayAmount)}
/>
)
})}
</div>
</div>
)
}
export default BorrowTable