bbbdca6950
* 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
198 lines
5.2 KiB
TypeScript
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
|