Skip to content

Commit f72d959

Browse files
Merge pull request #583 from curvefi/feat/table-settings
feat: column visibility settings
2 parents c3148a1 + ba17a40 commit f72d959

30 files changed

+518
-176
lines changed

apps/lend/src/globalStyle.ts

-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import { createGlobalStyle } from 'styled-components'
2-
import { CURVE_ASSETS_URL } from '@/ui/utils'
32

43
const GlobalStyle = createGlobalStyle`
54
/* || GENERAL STYLES */
@@ -17,11 +16,6 @@ const GlobalStyle = createGlobalStyle`
1716
1817
color: var(--page--text-color);
1918
background-color: var(--page--background-color);
20-
/* background-image: url(${CURVE_ASSETS_URL + '/branding/curve-app-header.webp'});
21-
background-size: auto 400px;
22-
background-repeat: repeat-x;
23-
background-attachment: fixed; */
24-
//background-position-y: var(--header-height);
2519
2620
&.scrollSmooth {
2721
scroll-behavior: smooth;

apps/lend/src/pages/_document.tsx

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1-
import Document, { Html, Head, Main, NextScript, DocumentContext } from 'next/document'
1+
import Document, { DocumentContext, Head, Html, Main, NextScript } from 'next/document'
22
import { ServerStyleSheet } from 'styled-components'
33
import { RootCssProperties } from '@ui-kit/themes/typography'
4+
import { CURVE_LOGO_URL } from '@/ui/utils/utilsConstants'
45

56
const injectIpfsPrefix = `
67
(function () {
@@ -65,7 +66,7 @@ export default class CurveDocument extends Document {
6566
property="og:description"
6667
content="Curve-frontend is a user interface application designed to connect to Curve's deployment of smart contracts."
6768
/>
68-
<meta property="og:image" content="https://cdn.jsdelivr.net/gh/curvefi/curve-assets/branding/logo.png" />
69+
<meta property="og:image" content={CURVE_LOGO_URL} />
6970

7071
{/* Twitter */}
7172
<meta property="twitter:card" content="summary" />
@@ -75,7 +76,7 @@ export default class CurveDocument extends Document {
7576
property="twitter:description"
7677
content="Curve-frontend is a user interface application designed to connect to Curve's deployment of smart contracts."
7778
/>
78-
<meta property="twitter:image" content="https://cdn.jsdelivr.net/gh/curvefi/curve-assets/branding/logo.png" />
79+
<meta property="twitter:image" content={CURVE_LOGO_URL} />
7980

8081
<link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" />
8182
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png" />

apps/loan/src/components/PageLlamaMarkets/LendingMarketsFilters.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ export const LendingMarketsFilters = ({
4242
setColumnFilter: (id: string, value: unknown) => void
4343
data: LendingVault[]
4444
}) => (
45-
<Grid container spacing={Spacing.md} paddingBlock={Spacing.sm}>
45+
<Grid container spacing={Spacing.sm} paddingTop={Spacing.sm}>
4646
<Grid size={{ mobile: 12, tablet: 4 }}>
4747
<MultiSelectFilter
4848
field="blockchainId"

apps/loan/src/components/PageLlamaMarkets/LendingMarketsTable.tsx

+44-13
Original file line numberDiff line numberDiff line change
@@ -16,8 +16,13 @@ import {
1616
import { LendingMarketsFilters } from '@/components/PageLlamaMarkets/LendingMarketsFilters'
1717
import { useSortFromQueryString } from '@ui-kit/hooks/useSortFromQueryString'
1818
import { DeepKeys } from '@tanstack/table-core/build/lib/utils'
19+
import {
20+
isFeatureVisible,
21+
useVisibilitySettings,
22+
VisibilityGroup,
23+
} from '@ui-kit/shared/ui/TableVisibilitySettingsPopover'
1924

20-
const { ColumnWidth, Spacing, MinWidth, MaxWidth } = SizesAndSpaces
25+
const { ColumnWidth, Spacing, MaxWidth } = SizesAndSpaces
2126

2227
const columnHelper = createColumnHelper<LendingVault>()
2328

@@ -28,6 +33,8 @@ const hidden = (id: DeepKeys<LendingVault>) =>
2833
meta: { hidden: true },
2934
})
3035

36+
const [borrowChartId, lendChartId] = ['borrowChart', 'lendChart']
37+
3138
/** Columns for the lending markets table. */
3239
const columns = [
3340
columnHelper.accessor('assets', {
@@ -37,13 +44,13 @@ const columns = [
3744
}),
3845
columnHelper.accessor('rates.borrowApyPcent', {
3946
header: t`7D Borrow Rate`,
40-
cell: (c) => <LineGraphCell vault={c.row.original} type="borrow" />,
47+
cell: (c) => <LineGraphCell vault={c.row.original} type="borrow" showChart={isFeatureVisible(c, borrowChartId)} />,
4148
meta: { type: 'numeric' },
4249
size: ColumnWidth.md,
4350
}),
4451
columnHelper.accessor('rates.lendApyPcent', {
4552
header: t`7D Supply Yield`,
46-
cell: (c) => <LineGraphCell vault={c.row.original} type="lend" />,
53+
cell: (c) => <LineGraphCell vault={c.row.original} type="lend" showChart={isFeatureVisible(c, lendChartId)} />,
4754
meta: { type: 'numeric' },
4855
size: ColumnWidth.md,
4956
}),
@@ -59,13 +66,32 @@ const columns = [
5966
meta: { type: 'numeric' },
6067
size: ColumnWidth.sm,
6168
}),
69+
// following columns are used to configure and filter tanstack, but they are displayed together in PoolTitleCell
6270
hidden('blockchainId'),
6371
hidden('assets.collateral.symbol'),
6472
hidden('assets.borrowed.symbol'),
6573
] satisfies ColumnDef<LendingVault, any>[]
6674

6775
const DEFAULT_SORT = [{ id: 'totalSupplied.usdTotal', desc: true }]
6876

77+
const DEFAULT_VISIBILITY: VisibilityGroup[] = [
78+
{
79+
label: t`Markets`,
80+
options: [
81+
{ label: t`Available Liquidity`, id: 'totalSupplied.usdTotal', active: true, type: 'column' },
82+
{ label: t`Utilization`, id: 'utilizationPercent', active: true, type: 'column' },
83+
],
84+
},
85+
{
86+
label: t`Borrow`,
87+
options: [{ label: t`Chart`, id: borrowChartId, active: true, type: 'feature' }],
88+
},
89+
{
90+
label: t`Lend`,
91+
options: [{ label: t`Chart`, id: lendChartId, active: true, type: 'feature' }],
92+
},
93+
]
94+
6995
export const LendingMarketsTable = ({
7096
onReload,
7197
data,
@@ -76,6 +102,8 @@ export const LendingMarketsTable = ({
76102
headerHeight: string
77103
}) => {
78104
const [columnFilters, columnFiltersById, setColumnFilter] = useColumnFilters()
105+
const { columnSettings, columnVisibility, featureVisibility, toggleVisibility } =
106+
useVisibilitySettings(DEFAULT_VISIBILITY)
79107

80108
const [sorting, onSortingChange] = useSortFromQueryString(DEFAULT_SORT)
81109
const table = useReactTable({
@@ -84,7 +112,7 @@ export const LendingMarketsTable = ({
84112
getCoreRowModel: getCoreRowModel(),
85113
getSortedRowModel: getSortedRowModel(),
86114
getFilteredRowModel: getFilteredRowModel(),
87-
state: { sorting, columnFilters },
115+
state: { sorting, columnVisibility, featureVisibility, columnFilters },
88116
onSortingChange,
89117
maxMultiSortColCount: 3, // allow 3 columns to be sorted at once
90118
})
@@ -97,15 +125,18 @@ export const LendingMarketsTable = ({
97125
maxWidth: MaxWidth.table,
98126
}}
99127
>
100-
<TableFilters
101-
title={t`Llamalend Markets`}
102-
subtitle={t`Select a market to view more details`}
103-
onReload={onReload}
104-
learnMoreUrl="https://docs.curve.fi/lending/overview/"
105-
>
106-
<LendingMarketsFilters columnFilters={columnFiltersById} setColumnFilter={setColumnFilter} data={data} />
107-
</TableFilters>
108-
<DataTable table={table} headerHeight={headerHeight} rowHeight={'3xl'} emptyText={t`No markets found`} />
128+
<DataTable table={table} headerHeight={headerHeight} rowHeight="3xl" emptyText={t`No markets found`}>
129+
<TableFilters
130+
title={t`Llamalend Markets`}
131+
subtitle={t`Select a market to view more details`}
132+
onReload={onReload}
133+
learnMoreUrl="https://docs.curve.fi/lending/overview/"
134+
visibilityGroups={columnSettings}
135+
toggleVisibility={toggleVisibility}
136+
>
137+
<LendingMarketsFilters columnFilters={columnFiltersById} setColumnFilter={setColumnFilter} data={data} />
138+
</TableFilters>
139+
</DataTable>
109140
</Stack>
110141
)
111142
}

apps/loan/src/components/PageLlamaMarkets/cells/LineGraphCell.tsx

+59-20
Original file line numberDiff line numberDiff line change
@@ -1,51 +1,90 @@
11
import { LendingSnapshot, useLendingSnapshots } from '@/entities/lending'
22
import { LendingVault } from '@/entities/vaults'
3-
import { Line, LineChart } from 'recharts'
3+
import { Line, LineChart, YAxis } from 'recharts'
44
import { useTheme } from '@mui/material/styles'
55
import { DesignSystem } from '@ui-kit/themes/design'
6-
import Box from '@mui/material/Box'
76
import Stack from '@mui/material/Stack'
87
import Skeleton from '@mui/material/Skeleton'
8+
import Typography from '@mui/material/Typography'
9+
import { t } from '@lingui/macro'
10+
import { useMemo } from 'react'
11+
import { meanBy } from 'lodash'
12+
import Box from '@mui/material/Box'
913

1014
const graphSize = { width: 172, height: 48 }
1115

1216
type GraphType = 'borrow' | 'lend'
1317

18+
/**
19+
* Get the color for the line graph. Will be green if the last value is higher than the first, red if lower, and blue if equal.
20+
*/
1421
function getColor(design: DesignSystem, data: LendingSnapshot[], type: GraphType) {
1522
if (!data.length) return undefined
1623
const first = data[0][`${type}_apy`]
1724
const last = data[data.length - 1][`${type}_apy`]
1825
return design.Text.TextColors[last === first ? 'Info' : last < first ? 'Error' : 'Success']
1926
}
2027

21-
export const LineGraphCell = ({ vault, type }: { vault: LendingVault; type: GraphType }) => {
28+
/** Center the y-axis around the first value */
29+
const calculateDomain =
30+
(first: number) =>
31+
([dataMin, dataMax]: [number, number]): [number, number] => {
32+
const diff = Math.max(dataMax - first, first - dataMin)
33+
return [first - diff, first + diff]
34+
}
35+
36+
/**
37+
* Line graph cell that displays the average historical APY for a vault and a given type (borrow or lend).
38+
*/
39+
export const LineGraphCell = ({
40+
vault,
41+
type,
42+
showChart,
43+
}: {
44+
vault: LendingVault
45+
type: GraphType
46+
showChart: boolean // chart is hidden depending on the chart settings
47+
}) => {
2248
const { data: snapshots, isLoading } = useLendingSnapshots({
2349
blockchainId: vault.blockchainId,
2450
contractAddress: vault.controllerAddress,
2551
})
2652
const { design } = useTheme()
27-
const value = vault.rates[`${type}ApyPcent`]
28-
if (value == null) {
53+
const currentValue = vault.rates[`${type}ApyPcent`]
54+
const snapshotKey = `${type}_apy` as const
55+
56+
const rate = useMemo(
57+
() => (snapshots?.length ? meanBy(snapshots, (row) => row[snapshotKey]) : currentValue),
58+
[snapshots, currentValue, snapshotKey],
59+
)
60+
if (rate == null) {
2961
return '-'
3062
}
3163

3264
return (
3365
<Stack direction="row" alignItems="center" justifyContent="end" gap={3} data-testid={`line-graph-cell-${type}`}>
34-
{value.toPrecision(4)}%
35-
{snapshots?.length ? (
36-
<LineChart data={snapshots} {...graphSize}>
37-
<Line
38-
type="monotone"
39-
dataKey={`${type}_apy`}
40-
stroke={getColor(design, snapshots, type)}
41-
strokeWidth={1}
42-
dot={<></>}
43-
/>
44-
</LineChart>
45-
) : isLoading ? (
46-
<Skeleton {...graphSize} />
47-
) : (
48-
<Box sx={graphSize} />
66+
{rate.toPrecision(4)}%
67+
{showChart && (
68+
<Box data-testid={`line-graph-${type}`}>
69+
{snapshots?.length ? (
70+
<LineChart data={snapshots} {...graphSize} compact>
71+
<YAxis hide type="number" domain={calculateDomain(snapshots[0][snapshotKey])} />
72+
<Line
73+
type="monotone"
74+
dataKey={snapshotKey}
75+
stroke={getColor(design, snapshots, type)}
76+
strokeWidth={1}
77+
dot={<></>}
78+
/>
79+
</LineChart>
80+
) : isLoading ? (
81+
<Skeleton {...graphSize} />
82+
) : (
83+
<Typography sx={{ ...graphSize, alignContent: 'center', textAlign: 'left' }} variant="bodyXsBold">
84+
{t`No historical data`}
85+
</Typography>
86+
)}
87+
</Box>
4988
)}
5089
</Stack>
5190
)

apps/loan/src/components/PageLlamaMarkets/cells/PoolTitleCell/PoolTitleCell.tsx

+7-2
Original file line numberDiff line numberDiff line change
@@ -8,11 +8,16 @@ import Typography from '@mui/material/Typography'
88
import { PoolBadges } from '@/components/PageLlamaMarkets/cells/PoolTitleCell/PoolBadges'
99
import { PoolWarnings } from '@/components/PageLlamaMarkets/cells/PoolTitleCell/PoolWarnings'
1010
import { getImageBaseUrl } from '@/ui/utils'
11+
import { cleanColumnId } from '@ui-kit/shared/ui/TableVisibilitySettingsPopover'
1112

1213
const { Spacing } = SizesAndSpaces
1314

14-
export const PoolTitleCell = ({ getValue, row }: CellContext<LendingVault, LendingVault['assets']>) => {
15-
const coins = useMemo(() => Object.values(getValue()), [getValue])
15+
export const PoolTitleCell = ({ getValue, row, table }: CellContext<LendingVault, LendingVault['assets']>) => {
16+
const showCollateral = table.getColumn(cleanColumnId('assets.collateral.symbol'))!.getIsVisible()
17+
const coins = useMemo(() => {
18+
const { borrowed, collateral } = getValue()
19+
return showCollateral ? [collateral, borrowed] : [borrowed]
20+
}, [getValue, showCollateral])
1621
const { blockchainId } = row.original
1722
const imageBaseUrl = getImageBaseUrl(blockchainId)
1823
return (

apps/loan/src/components/PageLlamaMarkets/cells/index.ts

-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,3 @@ export * from './CompactUsdCell'
22
export * from './PoolTitleCell'
33
export * from './LineGraphCell'
44
export * from './UtilizationCell'
5-
export { LinearProgress } from '@ui-kit/shared/ui/LinearProgress'

apps/loan/src/components/PageLlamaMarkets/filters/MinimumSliderFilter.tsx

+4-3
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
import { Fragment, useMemo } from 'react'
1+
import { useMemo } from 'react'
22
import Select from '@mui/material/Select'
33
import Slider from '@mui/material/Slider'
44
import { DeepKeys } from '@tanstack/table-core/build/lib/utils'
55
import { get } from 'lodash'
66
import Stack from '@mui/material/Stack'
77
import Typography from '@mui/material/Typography'
8+
import { cleanColumnId } from '@ui-kit/shared/ui/TableVisibilitySettingsPopover'
89

910
/**
1011
* Get the maximum value from a field in an array of objects.
@@ -31,7 +32,7 @@ export const MinimumSliderFilter = <T extends unknown>({
3132
field: DeepKeys<T>
3233
format: (value: number) => string
3334
}) => {
34-
const id = field.replaceAll('.', '_')
35+
const id = cleanColumnId(field)
3536
const max = useMemo(() => getMaxValueFromData(data, field), [data, field])
3637
const [value] = (columnFilters[id] ?? [0, max]) as [number, number] // tanstack expects a [min, max] tuple
3738
return (
@@ -51,7 +52,7 @@ export const MinimumSliderFilter = <T extends unknown>({
5152
)}
5253
value="" // we actually don't use the value of the select, but it needs to be set to avoid a warning
5354
>
54-
<Stack paddingBlock={3} paddingInline={4} direction="row" spacing={6}>
55+
<Stack paddingBlock={3} paddingInline={4} direction="row" spacing={6} alignItems="center">
5556
<Typography>{format(0)}</Typography>
5657
<Slider
5758
data-testid={`slider-${id}`}

apps/loan/src/components/PageLlamaMarkets/filters/MultiSelectFilter.tsx

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
1-
import { Fragment, ReactNode, useMemo } from 'react'
1+
import { ReactNode, useMemo } from 'react'
22
import { get, identity, sortBy, sortedUniq } from 'lodash'
33
import Select from '@mui/material/Select'
44
import Typography from '@mui/material/Typography'
55
import MenuItem from '@mui/material/MenuItem'
66
import { DeepKeys } from '@tanstack/table-core/build/lib/utils'
7+
import { cleanColumnId } from '@ui-kit/shared/ui/TableVisibilitySettingsPopover'
78
import { SizesAndSpaces } from '@ui-kit/themes/design/1_sizes_spaces'
89

910
const { Spacing } = SizesAndSpaces
@@ -36,7 +37,7 @@ export const MultiSelectFilter = <T extends unknown>({
3637
renderItem?: (value: string) => ReactNode
3738
}) => {
3839
const options = useMemo(() => getSortedStrings(data, field), [data, field])
39-
const id = field.replaceAll('.', '_')
40+
const id = cleanColumnId(field)
4041
const value = (columnFilters[id] ?? []) as string[]
4142
return (
4243
<Select

apps/loan/src/globalStyle.ts

-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import { createGlobalStyle } from 'styled-components'
2-
import { CURVE_ASSETS_URL } from '@/ui/utils'
32

43
const GlobalStyle = createGlobalStyle`
54
/* || GENERAL STYLES */
@@ -17,11 +16,6 @@ const GlobalStyle = createGlobalStyle`
1716
1817
color: var(--page--text-color);
1918
background-color: var(--page--background-color);
20-
/* background-image: url(${CURVE_ASSETS_URL + '/branding/curve-app-header.webp'}); */
21-
/* background-size: auto 400px;
22-
background-repeat: repeat-x;
23-
background-attachment: fixed; */
24-
//background-position-y: var(--header-height);
2519
2620
&.scrollSmooth {
2721
scroll-behavior: smooth;

0 commit comments

Comments
 (0)