2022-09-26 23:01:13 +00:00
|
|
|
import clsx from 'clsx'
|
2022-06-24 17:14:20 +00:00
|
|
|
import { formatMoney } from 'common/util/format'
|
|
|
|
import { last } from 'lodash'
|
2022-08-28 23:03:00 +00:00
|
|
|
import { memo, useRef, useState } from 'react'
|
|
|
|
import { usePortfolioHistory } from 'web/hooks/use-portfolio-history'
|
|
|
|
import { Period } from 'web/lib/firebase/users'
|
2022-09-26 23:01:13 +00:00
|
|
|
import { PillButton } from '../buttons/pill-button'
|
2022-06-24 17:14:20 +00:00
|
|
|
import { Col } from '../layout/col'
|
|
|
|
import { Row } from '../layout/row'
|
|
|
|
import { PortfolioValueGraph } from './portfolio-value-graph'
|
|
|
|
|
|
|
|
export const PortfolioValueSection = memo(
|
2022-08-27 08:09:01 +00:00
|
|
|
function PortfolioValueSection(props: { userId: string }) {
|
|
|
|
const { userId } = props
|
2022-08-12 03:44:51 +00:00
|
|
|
|
2022-08-22 04:02:56 +00:00
|
|
|
const [portfolioPeriod, setPortfolioPeriod] = useState<Period>('weekly')
|
2022-08-28 23:03:00 +00:00
|
|
|
const portfolioHistory = usePortfolioHistory(userId, portfolioPeriod)
|
2022-09-29 18:53:43 +00:00
|
|
|
const [graphMode, setGraphMode] = useState<'profit' | 'value'>('profit')
|
2022-09-26 23:01:13 +00:00
|
|
|
const [graphDisplayNumber, setGraphDisplayNumber] = useState<
|
|
|
|
number | string | null
|
|
|
|
>(null)
|
|
|
|
const handleGraphDisplayChange = (num: string | number | null) => {
|
|
|
|
setGraphDisplayNumber(num)
|
|
|
|
}
|
2022-08-27 08:09:01 +00:00
|
|
|
|
2022-08-28 23:03:00 +00:00
|
|
|
// Remember the last defined portfolio history.
|
|
|
|
const portfolioRef = useRef(portfolioHistory)
|
|
|
|
if (portfolioHistory) portfolioRef.current = portfolioHistory
|
|
|
|
const currPortfolioHistory = portfolioRef.current
|
2022-06-24 17:14:20 +00:00
|
|
|
|
2022-08-28 23:03:00 +00:00
|
|
|
const lastPortfolioMetrics = last(currPortfolioHistory)
|
|
|
|
if (!currPortfolioHistory || !lastPortfolioMetrics) {
|
2022-07-19 09:54:05 +00:00
|
|
|
return <></>
|
|
|
|
}
|
|
|
|
|
2022-09-07 04:24:56 +00:00
|
|
|
const { balance, investmentValue, totalDeposits } = lastPortfolioMetrics
|
2022-08-27 08:09:01 +00:00
|
|
|
const totalValue = balance + investmentValue
|
2022-09-07 04:24:56 +00:00
|
|
|
const totalProfit = totalValue - totalDeposits
|
2022-06-24 17:14:20 +00:00
|
|
|
return (
|
2022-08-27 08:09:01 +00:00
|
|
|
<>
|
2022-09-26 23:01:13 +00:00
|
|
|
<Row className="mb-2 justify-between">
|
|
|
|
<Row className="gap-4 sm:gap-8">
|
|
|
|
<Col
|
|
|
|
className={clsx(
|
|
|
|
'cursor-pointer',
|
|
|
|
graphMode != 'profit'
|
|
|
|
? 'cursor-pointer opacity-40 hover:opacity-80'
|
|
|
|
: ''
|
|
|
|
)}
|
|
|
|
onClick={() => setGraphMode('profit')}
|
|
|
|
>
|
|
|
|
<div className="text-greyscale-6 text-xs sm:text-sm">Profit</div>
|
|
|
|
<div
|
|
|
|
className={clsx(
|
|
|
|
graphMode === 'profit'
|
|
|
|
? graphDisplayNumber
|
|
|
|
? graphDisplayNumber.toString().includes('-')
|
|
|
|
? 'text-red-600'
|
|
|
|
: 'text-teal-500'
|
|
|
|
: totalProfit > 0
|
|
|
|
? 'text-teal-500'
|
|
|
|
: 'text-red-600'
|
|
|
|
: totalProfit > 0
|
|
|
|
? 'text-teal-500'
|
|
|
|
: 'text-red-600',
|
|
|
|
'text-lg sm:text-xl'
|
|
|
|
)}
|
|
|
|
>
|
|
|
|
{graphMode === 'profit'
|
|
|
|
? graphDisplayNumber
|
|
|
|
? graphDisplayNumber
|
|
|
|
: formatMoney(totalProfit)
|
|
|
|
: formatMoney(totalProfit)}
|
|
|
|
</div>
|
|
|
|
</Col>
|
2022-09-29 18:53:43 +00:00
|
|
|
|
|
|
|
<Col
|
|
|
|
className={clsx(
|
|
|
|
'cursor-pointer',
|
|
|
|
graphMode != 'value' ? 'opacity-40 hover:opacity-80' : ''
|
|
|
|
)}
|
|
|
|
onClick={() => setGraphMode('value')}
|
|
|
|
>
|
|
|
|
<div className="text-greyscale-6 text-xs sm:text-sm">
|
|
|
|
Portfolio value
|
|
|
|
</div>
|
|
|
|
<div className={clsx('text-lg text-indigo-600 sm:text-xl')}>
|
|
|
|
{graphMode === 'value'
|
|
|
|
? graphDisplayNumber
|
|
|
|
? graphDisplayNumber
|
|
|
|
: formatMoney(totalValue)
|
|
|
|
: formatMoney(totalValue)}
|
|
|
|
</div>
|
|
|
|
</Col>
|
2022-09-26 23:01:13 +00:00
|
|
|
</Row>
|
2022-06-24 17:14:20 +00:00
|
|
|
</Row>
|
|
|
|
<PortfolioValueGraph
|
2022-08-28 23:03:00 +00:00
|
|
|
portfolioHistory={currPortfolioHistory}
|
2022-09-26 23:01:13 +00:00
|
|
|
mode={graphMode}
|
|
|
|
handleGraphDisplayChange={handleGraphDisplayChange}
|
2022-09-07 04:24:56 +00:00
|
|
|
/>
|
2022-09-26 23:01:13 +00:00
|
|
|
<PortfolioPeriodSelection
|
|
|
|
portfolioPeriod={portfolioPeriod}
|
|
|
|
setPortfolioPeriod={setPortfolioPeriod}
|
|
|
|
className="border-greyscale-2 mt-2 gap-4 border-b"
|
|
|
|
selectClassName="text-indigo-600 text-bold border-b border-indigo-600"
|
2022-06-24 17:14:20 +00:00
|
|
|
/>
|
2022-08-27 08:09:01 +00:00
|
|
|
</>
|
2022-06-24 17:14:20 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
2022-09-26 23:01:13 +00:00
|
|
|
|
|
|
|
export function PortfolioPeriodSelection(props: {
|
|
|
|
setPortfolioPeriod: (string: any) => void
|
|
|
|
portfolioPeriod: string
|
|
|
|
className?: string
|
|
|
|
selectClassName?: string
|
|
|
|
}) {
|
|
|
|
const { setPortfolioPeriod, portfolioPeriod, className, selectClassName } =
|
|
|
|
props
|
|
|
|
return (
|
|
|
|
<Row className={clsx(className, 'text-greyscale-4')}>
|
|
|
|
<button
|
|
|
|
className={clsx(portfolioPeriod === 'daily' ? selectClassName : '')}
|
|
|
|
onClick={() => setPortfolioPeriod('daily' as Period)}
|
|
|
|
>
|
|
|
|
1D
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
className={clsx(portfolioPeriod === 'weekly' ? selectClassName : '')}
|
|
|
|
onClick={() => setPortfolioPeriod('weekly' as Period)}
|
|
|
|
>
|
|
|
|
1W
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
className={clsx(portfolioPeriod === 'monthly' ? selectClassName : '')}
|
|
|
|
onClick={() => setPortfolioPeriod('monthly' as Period)}
|
|
|
|
>
|
|
|
|
1M
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
className={clsx(portfolioPeriod === 'allTime' ? selectClassName : '')}
|
|
|
|
onClick={() => setPortfolioPeriod('allTime' as Period)}
|
|
|
|
>
|
|
|
|
ALL
|
|
|
|
</button>
|
|
|
|
</Row>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function GraphToggle(props: {
|
|
|
|
setGraphMode: (mode: 'profit' | 'value') => void
|
|
|
|
graphMode: string
|
|
|
|
}) {
|
|
|
|
const { setGraphMode, graphMode } = props
|
|
|
|
return (
|
|
|
|
<Row className="relative mt-1 ml-1 items-center gap-1.5 sm:ml-0 sm:gap-2">
|
|
|
|
<PillButton
|
|
|
|
selected={graphMode === 'value'}
|
|
|
|
onSelect={() => {
|
|
|
|
setGraphMode('value')
|
|
|
|
}}
|
|
|
|
xs={true}
|
|
|
|
className="z-50"
|
|
|
|
>
|
|
|
|
Value
|
|
|
|
</PillButton>
|
|
|
|
<PillButton
|
|
|
|
selected={graphMode === 'profit'}
|
|
|
|
onSelect={() => {
|
|
|
|
setGraphMode('profit')
|
|
|
|
}}
|
|
|
|
xs={true}
|
|
|
|
className="z-50"
|
|
|
|
>
|
|
|
|
Profit
|
|
|
|
</PillButton>
|
|
|
|
</Row>
|
|
|
|
)
|
|
|
|
}
|