Inga/profile (#937)
- Changed edit profile button - got rid of banner - merged stats and trades tab on profile - made multicolored profit graph
This commit is contained in:
parent
d612192109
commit
2fe9fe593d
|
@ -77,7 +77,7 @@ export function BetsList(props: { user: User }) {
|
||||||
}, [contractList])
|
}, [contractList])
|
||||||
|
|
||||||
const [sort, setSort] = useState<BetSort>('newest')
|
const [sort, setSort] = useState<BetSort>('newest')
|
||||||
const [filter, setFilter] = useState<BetFilter>('open')
|
const [filter, setFilter] = useState<BetFilter>('all')
|
||||||
const [page, setPage] = useState(0)
|
const [page, setPage] = useState(0)
|
||||||
const start = page * CONTRACTS_PER_PAGE
|
const start = page * CONTRACTS_PER_PAGE
|
||||||
const end = start + CONTRACTS_PER_PAGE
|
const end = start + CONTRACTS_PER_PAGE
|
||||||
|
@ -155,34 +155,25 @@ export function BetsList(props: { user: User }) {
|
||||||
(c) => contractsMetrics[c.id].netPayout
|
(c) => contractsMetrics[c.id].netPayout
|
||||||
)
|
)
|
||||||
|
|
||||||
const totalPnl = user.profitCached.allTime
|
|
||||||
const totalProfitPercent = (totalPnl / user.totalDeposits) * 100
|
|
||||||
const investedProfitPercent =
|
const investedProfitPercent =
|
||||||
((currentBetsValue - currentInvested) / (currentInvested + 0.1)) * 100
|
((currentBetsValue - currentInvested) / (currentInvested + 0.1)) * 100
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Col>
|
<Col>
|
||||||
<Col className="mx-4 gap-4 sm:flex-row sm:justify-between md:mx-0">
|
<Row className="justify-between gap-4 sm:flex-row">
|
||||||
<Row className="gap-8">
|
|
||||||
<Col>
|
<Col>
|
||||||
<div className="text-sm text-gray-500">Investment value</div>
|
<div className="text-greyscale-6 text-xs sm:text-sm">
|
||||||
|
Investment value
|
||||||
|
</div>
|
||||||
<div className="text-lg">
|
<div className="text-lg">
|
||||||
{formatMoney(currentNetInvestment)}{' '}
|
{formatMoney(currentNetInvestment)}{' '}
|
||||||
<ProfitBadge profitPercent={investedProfitPercent} />
|
<ProfitBadge profitPercent={investedProfitPercent} />
|
||||||
</div>
|
</div>
|
||||||
</Col>
|
</Col>
|
||||||
<Col>
|
|
||||||
<div className="text-sm text-gray-500">Total profit</div>
|
|
||||||
<div className="text-lg">
|
|
||||||
{formatMoney(totalPnl)}{' '}
|
|
||||||
<ProfitBadge profitPercent={totalProfitPercent} />
|
|
||||||
</div>
|
|
||||||
</Col>
|
|
||||||
</Row>
|
|
||||||
|
|
||||||
<Row className="gap-8">
|
<Row className="gap-2">
|
||||||
<select
|
<select
|
||||||
className="select select-bordered self-start"
|
className="border-greyscale-4 self-start overflow-hidden rounded border px-2 py-2 text-sm"
|
||||||
value={filter}
|
value={filter}
|
||||||
onChange={(e) => setFilter(e.target.value as BetFilter)}
|
onChange={(e) => setFilter(e.target.value as BetFilter)}
|
||||||
>
|
>
|
||||||
|
@ -195,7 +186,7 @@ export function BetsList(props: { user: User }) {
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<select
|
<select
|
||||||
className="select select-bordered self-start"
|
className="border-greyscale-4 self-start overflow-hidden rounded px-2 py-2 text-sm"
|
||||||
value={sort}
|
value={sort}
|
||||||
onChange={(e) => setSort(e.target.value as BetSort)}
|
onChange={(e) => setSort(e.target.value as BetSort)}
|
||||||
>
|
>
|
||||||
|
@ -205,7 +196,7 @@ export function BetsList(props: { user: User }) {
|
||||||
<option value="closeTime">Close date</option>
|
<option value="closeTime">Close date</option>
|
||||||
</select>
|
</select>
|
||||||
</Row>
|
</Row>
|
||||||
</Col>
|
</Row>
|
||||||
|
|
||||||
<Col className="mt-6 divide-y">
|
<Col className="mt-6 divide-y">
|
||||||
{displayedContracts.length === 0 ? (
|
{displayedContracts.length === 0 ? (
|
||||||
|
|
|
@ -103,6 +103,7 @@ export function ContractSearch(props: {
|
||||||
loadMore: () => void
|
loadMore: () => void
|
||||||
) => ReactNode
|
) => ReactNode
|
||||||
autoFocus?: boolean
|
autoFocus?: boolean
|
||||||
|
profile?: boolean | undefined
|
||||||
}) {
|
}) {
|
||||||
const {
|
const {
|
||||||
user,
|
user,
|
||||||
|
@ -123,6 +124,7 @@ export function ContractSearch(props: {
|
||||||
maxResults,
|
maxResults,
|
||||||
renderContracts,
|
renderContracts,
|
||||||
autoFocus,
|
autoFocus,
|
||||||
|
profile,
|
||||||
} = props
|
} = props
|
||||||
|
|
||||||
const [state, setState] = usePersistentState(
|
const [state, setState] = usePersistentState(
|
||||||
|
@ -239,6 +241,10 @@ export function ContractSearch(props: {
|
||||||
/>
|
/>
|
||||||
{renderContracts ? (
|
{renderContracts ? (
|
||||||
renderContracts(renderedContracts, performQuery)
|
renderContracts(renderedContracts, performQuery)
|
||||||
|
) : renderedContracts && renderedContracts.length === 0 && profile ? (
|
||||||
|
<p className="mx-2 text-gray-500">
|
||||||
|
This creator does not yet have any markets.
|
||||||
|
</p>
|
||||||
) : (
|
) : (
|
||||||
<ContractsGrid
|
<ContractsGrid
|
||||||
contracts={renderedContracts}
|
contracts={renderedContracts}
|
||||||
|
|
|
@ -128,6 +128,7 @@ export function CreatorContractsList(props: {
|
||||||
creatorId: creator.id,
|
creatorId: creator.id,
|
||||||
}}
|
}}
|
||||||
persistPrefix={`user-${creator.id}`}
|
persistPrefix={`user-${creator.id}`}
|
||||||
|
profile={true}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,16 +13,18 @@ import { useDiscoverUsers } from 'web/hooks/use-users'
|
||||||
import { TextButton } from './text-button'
|
import { TextButton } from './text-button'
|
||||||
import { track } from 'web/lib/service/analytics'
|
import { track } from 'web/lib/service/analytics'
|
||||||
|
|
||||||
export function FollowingButton(props: { user: User }) {
|
export function FollowingButton(props: { user: User; className?: string }) {
|
||||||
const { user } = props
|
const { user, className } = props
|
||||||
const [isOpen, setIsOpen] = useState(false)
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
const followingIds = useFollows(user.id)
|
const followingIds = useFollows(user.id)
|
||||||
const followerIds = useFollowers(user.id)
|
const followerIds = useFollowers(user.id)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<TextButton onClick={() => setIsOpen(true)}>
|
<TextButton onClick={() => setIsOpen(true)} className={className}>
|
||||||
<span className="font-semibold">{followingIds?.length ?? ''}</span>{' '}
|
<span className={clsx('font-semibold')}>
|
||||||
|
{followingIds?.length ?? ''}
|
||||||
|
</span>{' '}
|
||||||
Following
|
Following
|
||||||
</TextButton>
|
</TextButton>
|
||||||
|
|
||||||
|
@ -69,15 +71,15 @@ export function EditFollowingButton(props: { user: User; className?: string }) {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export function FollowersButton(props: { user: User }) {
|
export function FollowersButton(props: { user: User; className?: string }) {
|
||||||
const { user } = props
|
const { user, className } = props
|
||||||
const [isOpen, setIsOpen] = useState(false)
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
const followingIds = useFollows(user.id)
|
const followingIds = useFollows(user.id)
|
||||||
const followerIds = useFollowers(user.id)
|
const followerIds = useFollowers(user.id)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<TextButton onClick={() => setIsOpen(true)}>
|
<TextButton onClick={() => setIsOpen(true)} className={className}>
|
||||||
<span className="font-semibold">{followerIds?.length ?? ''}</span>{' '}
|
<span className="font-semibold">{followerIds?.length ?? ''}</span>{' '}
|
||||||
Followers
|
Followers
|
||||||
</TextButton>
|
</TextButton>
|
||||||
|
|
|
@ -14,14 +14,14 @@ import { firebaseLogin } from 'web/lib/firebase/users'
|
||||||
import { GroupLinkItem } from 'web/pages/groups'
|
import { GroupLinkItem } from 'web/pages/groups'
|
||||||
import toast from 'react-hot-toast'
|
import toast from 'react-hot-toast'
|
||||||
|
|
||||||
export function GroupsButton(props: { user: User }) {
|
export function GroupsButton(props: { user: User; className?: string }) {
|
||||||
const { user } = props
|
const { user, className } = props
|
||||||
const [isOpen, setIsOpen] = useState(false)
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
const groups = useMemberGroups(user.id)
|
const groups = useMemberGroups(user.id)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<TextButton onClick={() => setIsOpen(true)}>
|
<TextButton onClick={() => setIsOpen(true)} className={className}>
|
||||||
<span className="font-semibold">{groups?.length ?? ''}</span> Groups
|
<span className="font-semibold">{groups?.length ?? ''}</span> Groups
|
||||||
</TextButton>
|
</TextButton>
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,7 @@ import clsx from 'clsx'
|
||||||
import { useRouter, NextRouter } from 'next/router'
|
import { useRouter, NextRouter } from 'next/router'
|
||||||
import { ReactNode, useState } from 'react'
|
import { ReactNode, useState } from 'react'
|
||||||
import { track } from '@amplitude/analytics-browser'
|
import { track } from '@amplitude/analytics-browser'
|
||||||
|
import { Col } from './col'
|
||||||
|
|
||||||
type Tab = {
|
type Tab = {
|
||||||
title: string
|
title: string
|
||||||
|
@ -55,11 +56,13 @@ export function ControlledTabs(props: TabProps & { activeIndex: number }) {
|
||||||
)}
|
)}
|
||||||
aria-current={activeIndex === i ? 'page' : undefined}
|
aria-current={activeIndex === i ? 'page' : undefined}
|
||||||
>
|
>
|
||||||
{tab.tabIcon && <span>{tab.tabIcon}</span>}
|
|
||||||
{tab.badge ? (
|
{tab.badge ? (
|
||||||
<span className="px-0.5 font-bold">{tab.badge}</span>
|
<span className="px-0.5 font-bold">{tab.badge}</span>
|
||||||
) : null}
|
) : null}
|
||||||
|
<Col>
|
||||||
|
{tab.tabIcon && <div className="mx-auto">{tab.tabIcon}</div>}
|
||||||
{tab.title}
|
{tab.title}
|
||||||
|
</Col>
|
||||||
</a>
|
</a>
|
||||||
))}
|
))}
|
||||||
</nav>
|
</nav>
|
||||||
|
|
|
@ -1,72 +1,155 @@
|
||||||
import { ResponsiveLine } from '@nivo/line'
|
import { ResponsiveLine } from '@nivo/line'
|
||||||
import { PortfolioMetrics } from 'common/user'
|
import { PortfolioMetrics } from 'common/user'
|
||||||
|
import { filterDefined } from 'common/util/array'
|
||||||
import { formatMoney } from 'common/util/format'
|
import { formatMoney } from 'common/util/format'
|
||||||
|
import dayjs from 'dayjs'
|
||||||
import { last } from 'lodash'
|
import { last } from 'lodash'
|
||||||
import { memo } from 'react'
|
import { memo } from 'react'
|
||||||
import { useWindowSize } from 'web/hooks/use-window-size'
|
import { useWindowSize } from 'web/hooks/use-window-size'
|
||||||
import { formatTime } from 'web/lib/util/time'
|
import { Col } from '../layout/col'
|
||||||
|
|
||||||
export const PortfolioValueGraph = memo(function PortfolioValueGraph(props: {
|
export const PortfolioValueGraph = memo(function PortfolioValueGraph(props: {
|
||||||
portfolioHistory: PortfolioMetrics[]
|
portfolioHistory: PortfolioMetrics[]
|
||||||
mode: 'value' | 'profit'
|
mode: 'value' | 'profit'
|
||||||
|
handleGraphDisplayChange: (arg0: string | number | null) => void
|
||||||
height?: number
|
height?: number
|
||||||
includeTime?: boolean
|
|
||||||
}) {
|
}) {
|
||||||
const { portfolioHistory, height, includeTime, mode } = props
|
const { portfolioHistory, height, mode, handleGraphDisplayChange } = props
|
||||||
const { width } = useWindowSize()
|
const { width } = useWindowSize()
|
||||||
|
|
||||||
const points = portfolioHistory.map((p) => {
|
const valuePoints = getPoints('value', portfolioHistory)
|
||||||
const { timestamp, balance, investmentValue, totalDeposits } = p
|
const posProfitPoints = getPoints('posProfit', portfolioHistory)
|
||||||
const value = balance + investmentValue
|
const negProfitPoints = getPoints('negProfit', portfolioHistory)
|
||||||
const profit = value - totalDeposits
|
|
||||||
|
|
||||||
return {
|
const valuePointsY = valuePoints.map((p) => p.y)
|
||||||
x: new Date(timestamp),
|
const posProfitPointsY = posProfitPoints.map((p) => p.y)
|
||||||
y: mode === 'value' ? value : profit,
|
const negProfitPointsY = negProfitPoints.map((p) => p.y)
|
||||||
|
|
||||||
|
let data
|
||||||
|
|
||||||
|
if (mode === 'value') {
|
||||||
|
data = [{ id: 'value', data: valuePoints, color: '#4f46e5' }]
|
||||||
|
} else {
|
||||||
|
data = [
|
||||||
|
{
|
||||||
|
id: 'negProfit',
|
||||||
|
data: negProfitPoints,
|
||||||
|
color: '#dc2626',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'posProfit',
|
||||||
|
data: posProfitPoints,
|
||||||
|
color: '#14b8a6',
|
||||||
|
},
|
||||||
|
]
|
||||||
}
|
}
|
||||||
})
|
const numYTickValues = 2
|
||||||
const data = [{ id: 'Value', data: points, color: '#11b981' }]
|
const endDate = last(data[0].data)?.x
|
||||||
const numXTickValues = !width || width < 800 ? 2 : 5
|
|
||||||
const numYTickValues = 4
|
const yMin =
|
||||||
const endDate = last(points)?.x
|
mode === 'value'
|
||||||
|
? Math.min(...filterDefined(valuePointsY))
|
||||||
|
: Math.min(
|
||||||
|
...filterDefined(negProfitPointsY),
|
||||||
|
...filterDefined(posProfitPointsY)
|
||||||
|
)
|
||||||
|
|
||||||
|
const yMax =
|
||||||
|
mode === 'value'
|
||||||
|
? Math.max(...filterDefined(valuePointsY))
|
||||||
|
: Math.max(
|
||||||
|
...filterDefined(negProfitPointsY),
|
||||||
|
...filterDefined(posProfitPointsY)
|
||||||
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className="w-full overflow-hidden"
|
className="w-full overflow-hidden"
|
||||||
style={{ height: height ?? (!width || width >= 800 ? 350 : 250) }}
|
style={{ height: height ?? (!width || width >= 800 ? 200 : 100) }}
|
||||||
|
onMouseLeave={() => handleGraphDisplayChange(null)}
|
||||||
>
|
>
|
||||||
<ResponsiveLine
|
<ResponsiveLine
|
||||||
|
margin={{ top: 10, right: 0, left: 40, bottom: 10 }}
|
||||||
data={data}
|
data={data}
|
||||||
margin={{ top: 20, right: 28, bottom: 22, left: 60 }}
|
|
||||||
xScale={{
|
xScale={{
|
||||||
type: 'time',
|
type: 'time',
|
||||||
min: points[0]?.x,
|
min: valuePoints[0]?.x,
|
||||||
max: endDate,
|
max: endDate,
|
||||||
}}
|
}}
|
||||||
yScale={{
|
yScale={{
|
||||||
type: 'linear',
|
type: 'linear',
|
||||||
stacked: false,
|
stacked: false,
|
||||||
min: Math.min(...points.map((p) => p.y)),
|
min: yMin,
|
||||||
|
max: yMax,
|
||||||
}}
|
}}
|
||||||
gridYValues={numYTickValues}
|
|
||||||
curve="stepAfter"
|
curve="stepAfter"
|
||||||
enablePoints={false}
|
enablePoints={false}
|
||||||
colors={{ datum: 'color' }}
|
colors={{ datum: 'color' }}
|
||||||
axisBottom={{
|
axisBottom={{
|
||||||
tickValues: numXTickValues,
|
tickValues: 0,
|
||||||
format: (time) => formatTime(+time, !!includeTime),
|
|
||||||
}}
|
}}
|
||||||
pointBorderColor="#fff"
|
pointBorderColor="#fff"
|
||||||
pointSize={points.length > 100 ? 0 : 6}
|
pointSize={valuePoints.length > 100 ? 0 : 6}
|
||||||
axisLeft={{
|
axisLeft={{
|
||||||
tickValues: numYTickValues,
|
tickValues: numYTickValues,
|
||||||
format: (value) => formatMoney(value),
|
format: '.3s',
|
||||||
}}
|
}}
|
||||||
enableGridX={!!width && width >= 800}
|
enableGridX={false}
|
||||||
enableGridY={true}
|
enableGridY={true}
|
||||||
|
gridYValues={numYTickValues}
|
||||||
enableSlices="x"
|
enableSlices="x"
|
||||||
animate={false}
|
animate={false}
|
||||||
yFormat={(value) => formatMoney(+value)}
|
yFormat={(value) => formatMoney(+value)}
|
||||||
|
enableArea={true}
|
||||||
|
areaOpacity={0.1}
|
||||||
|
sliceTooltip={({ slice }) => {
|
||||||
|
handleGraphDisplayChange(slice.points[0].data.yFormatted)
|
||||||
|
return (
|
||||||
|
<div className="rounded bg-white px-4 py-2 opacity-80">
|
||||||
|
<div
|
||||||
|
key={slice.points[0].id}
|
||||||
|
className="text-xs font-semibold sm:text-sm"
|
||||||
|
>
|
||||||
|
<Col>
|
||||||
|
<div>
|
||||||
|
{dayjs(slice.points[0].data.xFormatted).format('MMM/D/YY')}
|
||||||
|
</div>
|
||||||
|
<div className="text-greyscale-6 text-2xs font-normal sm:text-xs">
|
||||||
|
{dayjs(slice.points[0].data.xFormatted).format('h:mm A')}
|
||||||
|
</div>
|
||||||
|
</Col>
|
||||||
|
</div>
|
||||||
|
{/* ))} */}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}}
|
||||||
></ResponsiveLine>
|
></ResponsiveLine>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export function getPoints(
|
||||||
|
line: 'value' | 'posProfit' | 'negProfit',
|
||||||
|
portfolioHistory: PortfolioMetrics[]
|
||||||
|
) {
|
||||||
|
const points = portfolioHistory.map((p) => {
|
||||||
|
const { timestamp, balance, investmentValue, totalDeposits } = p
|
||||||
|
const value = balance + investmentValue
|
||||||
|
|
||||||
|
const profit = value - totalDeposits
|
||||||
|
let posProfit = null
|
||||||
|
let negProfit = null
|
||||||
|
if (profit < 0) {
|
||||||
|
negProfit = profit
|
||||||
|
} else {
|
||||||
|
posProfit = profit
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
x: new Date(timestamp),
|
||||||
|
y:
|
||||||
|
line === 'value' ? value : line === 'posProfit' ? posProfit : negProfit,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return points
|
||||||
|
}
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
|
import clsx from 'clsx'
|
||||||
import { formatMoney } from 'common/util/format'
|
import { formatMoney } from 'common/util/format'
|
||||||
import { last } from 'lodash'
|
import { last } from 'lodash'
|
||||||
import { memo, useRef, useState } from 'react'
|
import { memo, useRef, useState } from 'react'
|
||||||
import { usePortfolioHistory } from 'web/hooks/use-portfolio-history'
|
import { usePortfolioHistory } from 'web/hooks/use-portfolio-history'
|
||||||
import { Period } from 'web/lib/firebase/users'
|
import { Period } from 'web/lib/firebase/users'
|
||||||
|
import { PillButton } from '../buttons/pill-button'
|
||||||
import { Col } from '../layout/col'
|
import { Col } from '../layout/col'
|
||||||
import { Row } from '../layout/row'
|
import { Row } from '../layout/row'
|
||||||
import { Spacer } from '../layout/spacer'
|
|
||||||
import { PortfolioValueGraph } from './portfolio-value-graph'
|
import { PortfolioValueGraph } from './portfolio-value-graph'
|
||||||
|
|
||||||
export const PortfolioValueSection = memo(
|
export const PortfolioValueSection = memo(
|
||||||
|
@ -14,6 +15,13 @@ export const PortfolioValueSection = memo(
|
||||||
|
|
||||||
const [portfolioPeriod, setPortfolioPeriod] = useState<Period>('weekly')
|
const [portfolioPeriod, setPortfolioPeriod] = useState<Period>('weekly')
|
||||||
const portfolioHistory = usePortfolioHistory(userId, portfolioPeriod)
|
const portfolioHistory = usePortfolioHistory(userId, portfolioPeriod)
|
||||||
|
const [graphMode, setGraphMode] = useState<'profit' | 'value'>('value')
|
||||||
|
const [graphDisplayNumber, setGraphDisplayNumber] = useState<
|
||||||
|
number | string | null
|
||||||
|
>(null)
|
||||||
|
const handleGraphDisplayChange = (num: string | number | null) => {
|
||||||
|
setGraphDisplayNumber(num)
|
||||||
|
}
|
||||||
|
|
||||||
// Remember the last defined portfolio history.
|
// Remember the last defined portfolio history.
|
||||||
const portfolioRef = useRef(portfolioHistory)
|
const portfolioRef = useRef(portfolioHistory)
|
||||||
|
@ -28,43 +36,144 @@ export const PortfolioValueSection = memo(
|
||||||
const { balance, investmentValue, totalDeposits } = lastPortfolioMetrics
|
const { balance, investmentValue, totalDeposits } = lastPortfolioMetrics
|
||||||
const totalValue = balance + investmentValue
|
const totalValue = balance + investmentValue
|
||||||
const totalProfit = totalValue - totalDeposits
|
const totalProfit = totalValue - totalDeposits
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Row className="gap-8">
|
<Row className="mb-2 justify-between">
|
||||||
<Col className="flex-1 justify-center">
|
<Row className="gap-4 sm:gap-8">
|
||||||
<div className="text-sm text-gray-500">Profit</div>
|
<Col
|
||||||
<div className="text-lg">{formatMoney(totalProfit)}</div>
|
className={clsx(
|
||||||
</Col>
|
'cursor-pointer',
|
||||||
<select
|
graphMode != 'value' ? 'opacity-40 hover:opacity-80' : ''
|
||||||
className="select select-bordered self-start"
|
)}
|
||||||
value={portfolioPeriod}
|
onClick={() => setGraphMode('value')}
|
||||||
onChange={(e) => {
|
|
||||||
setPortfolioPeriod(e.target.value as Period)
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<option value="allTime">All time</option>
|
<div className="text-greyscale-6 text-xs sm:text-sm">
|
||||||
<option value="monthly">Last Month</option>
|
Portfolio value
|
||||||
<option value="weekly">Last 7d</option>
|
</div>
|
||||||
<option value="daily">Last 24h</option>
|
<div className={clsx('text-lg text-indigo-600 sm:text-xl')}>
|
||||||
</select>
|
{graphMode === 'value'
|
||||||
|
? graphDisplayNumber
|
||||||
|
? graphDisplayNumber
|
||||||
|
: formatMoney(totalValue)
|
||||||
|
: formatMoney(totalValue)}
|
||||||
|
</div>
|
||||||
|
</Col>
|
||||||
|
<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>
|
||||||
|
</Row>
|
||||||
</Row>
|
</Row>
|
||||||
<PortfolioValueGraph
|
<PortfolioValueGraph
|
||||||
portfolioHistory={currPortfolioHistory}
|
portfolioHistory={currPortfolioHistory}
|
||||||
includeTime={portfolioPeriod == 'daily'}
|
mode={graphMode}
|
||||||
mode="profit"
|
handleGraphDisplayChange={handleGraphDisplayChange}
|
||||||
/>
|
/>
|
||||||
<Spacer h={8} />
|
<PortfolioPeriodSelection
|
||||||
<Col className="flex-1 justify-center">
|
portfolioPeriod={portfolioPeriod}
|
||||||
<div className="text-sm text-gray-500">Portfolio value</div>
|
setPortfolioPeriod={setPortfolioPeriod}
|
||||||
<div className="text-lg">{formatMoney(totalValue)}</div>
|
className="border-greyscale-2 mt-2 gap-4 border-b"
|
||||||
</Col>
|
selectClassName="text-indigo-600 text-bold border-b border-indigo-600"
|
||||||
<PortfolioValueGraph
|
|
||||||
portfolioHistory={currPortfolioHistory}
|
|
||||||
includeTime={portfolioPeriod == 'daily'}
|
|
||||||
mode="value"
|
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
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>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -10,15 +10,15 @@ import { XIcon } from '@heroicons/react/outline'
|
||||||
import { unLikeContract } from 'web/lib/firebase/likes'
|
import { unLikeContract } from 'web/lib/firebase/likes'
|
||||||
import { contractPath } from 'web/lib/firebase/contracts'
|
import { contractPath } from 'web/lib/firebase/contracts'
|
||||||
|
|
||||||
export function UserLikesButton(props: { user: User }) {
|
export function UserLikesButton(props: { user: User; className?: string }) {
|
||||||
const { user } = props
|
const { user, className } = props
|
||||||
const [isOpen, setIsOpen] = useState(false)
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
|
|
||||||
const likedContracts = useUserLikedContracts(user.id)
|
const likedContracts = useUserLikedContracts(user.id)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<TextButton onClick={() => setIsOpen(true)}>
|
<TextButton onClick={() => setIsOpen(true)} className={className}>
|
||||||
<span className="font-semibold">{likedContracts?.length ?? ''}</span>{' '}
|
<span className="font-semibold">{likedContracts?.length ?? ''}</span>{' '}
|
||||||
Likes
|
Likes
|
||||||
</TextButton>
|
</TextButton>
|
||||||
|
|
|
@ -13,14 +13,18 @@ import { getUser, updateUser } from 'web/lib/firebase/users'
|
||||||
import { TextButton } from 'web/components/text-button'
|
import { TextButton } from 'web/components/text-button'
|
||||||
import { UserLink } from 'web/components/user-link'
|
import { UserLink } from 'web/components/user-link'
|
||||||
|
|
||||||
export function ReferralsButton(props: { user: User; currentUser?: User }) {
|
export function ReferralsButton(props: {
|
||||||
const { user, currentUser } = props
|
user: User
|
||||||
|
currentUser?: User
|
||||||
|
className?: string
|
||||||
|
}) {
|
||||||
|
const { user, currentUser, className } = props
|
||||||
const [isOpen, setIsOpen] = useState(false)
|
const [isOpen, setIsOpen] = useState(false)
|
||||||
const referralIds = useReferrals(user.id)
|
const referralIds = useReferrals(user.id)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<TextButton onClick={() => setIsOpen(true)}>
|
<TextButton onClick={() => setIsOpen(true)} className={className}>
|
||||||
<span className="font-semibold">{referralIds?.length ?? ''}</span>{' '}
|
<span className="font-semibold">{referralIds?.length ?? ''}</span>{' '}
|
||||||
Referrals
|
Referrals
|
||||||
</TextButton>
|
</TextButton>
|
||||||
|
|
|
@ -1,8 +1,13 @@
|
||||||
import clsx from 'clsx'
|
import clsx from 'clsx'
|
||||||
import { useEffect, useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
import { useRouter } from 'next/router'
|
import { NextRouter, useRouter } from 'next/router'
|
||||||
import { LinkIcon } from '@heroicons/react/solid'
|
import { LinkIcon } from '@heroicons/react/solid'
|
||||||
import { PencilIcon } from '@heroicons/react/outline'
|
import {
|
||||||
|
ChatIcon,
|
||||||
|
FolderIcon,
|
||||||
|
PencilIcon,
|
||||||
|
ScaleIcon,
|
||||||
|
} from '@heroicons/react/outline'
|
||||||
|
|
||||||
import { User } from 'web/lib/firebase/users'
|
import { User } from 'web/lib/firebase/users'
|
||||||
import { useUser } from 'web/hooks/use-user'
|
import { useUser } from 'web/hooks/use-user'
|
||||||
|
@ -24,39 +29,23 @@ import { FollowersButton, FollowingButton } from './following-button'
|
||||||
import { UserFollowButton } from './follow-button'
|
import { UserFollowButton } from './follow-button'
|
||||||
import { GroupsButton } from 'web/components/groups/groups-button'
|
import { GroupsButton } from 'web/components/groups/groups-button'
|
||||||
import { PortfolioValueSection } from './portfolio/portfolio-value-section'
|
import { PortfolioValueSection } from './portfolio/portfolio-value-section'
|
||||||
import { ReferralsButton } from 'web/components/referrals-button'
|
|
||||||
import { formatMoney } from 'common/util/format'
|
import { formatMoney } from 'common/util/format'
|
||||||
import { ShareIconButton } from 'web/components/share-icon-button'
|
|
||||||
import { ENV_CONFIG } from 'common/envs/constants'
|
|
||||||
import {
|
import {
|
||||||
BettingStreakModal,
|
BettingStreakModal,
|
||||||
hasCompletedStreakToday,
|
hasCompletedStreakToday,
|
||||||
} from 'web/components/profile/betting-streak-modal'
|
} from 'web/components/profile/betting-streak-modal'
|
||||||
import { REFERRAL_AMOUNT } from 'common/economy'
|
|
||||||
import { LoansModal } from './profile/loans-modal'
|
import { LoansModal } from './profile/loans-modal'
|
||||||
import { UserLikesButton } from 'web/components/profile/user-likes-button'
|
|
||||||
import { PAST_BETS } from 'common/user'
|
|
||||||
import { capitalize } from 'lodash'
|
|
||||||
|
|
||||||
export function UserPage(props: { user: User }) {
|
export function UserPage(props: { user: User }) {
|
||||||
const { user } = props
|
const { user } = props
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const currentUser = useUser()
|
const currentUser = useUser()
|
||||||
const isCurrentUser = user.id === currentUser?.id
|
const isCurrentUser = user.id === currentUser?.id
|
||||||
const bannerUrl = user.bannerUrl ?? defaultBannerUrl(user.id)
|
|
||||||
const [showConfetti, setShowConfetti] = useState(false)
|
const [showConfetti, setShowConfetti] = useState(false)
|
||||||
const [showBettingStreakModal, setShowBettingStreakModal] = useState(false)
|
|
||||||
const [showLoansModal, setShowLoansModal] = useState(false)
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const claimedMana = router.query['claimed-mana'] === 'yes'
|
const claimedMana = router.query['claimed-mana'] === 'yes'
|
||||||
const showBettingStreak = router.query['show'] === 'betting-streak'
|
setShowConfetti(claimedMana)
|
||||||
setShowBettingStreakModal(showBettingStreak)
|
|
||||||
setShowConfetti(claimedMana || showBettingStreak)
|
|
||||||
|
|
||||||
const showLoansModel = router.query['show'] === 'loans'
|
|
||||||
setShowLoansModal(showLoansModel)
|
|
||||||
|
|
||||||
const query = { ...router.query }
|
const query = { ...router.query }
|
||||||
if (query.claimedMana || query.show) {
|
if (query.claimedMana || query.show) {
|
||||||
delete query['claimed-mana']
|
delete query['claimed-mana']
|
||||||
|
@ -85,102 +74,65 @@ export function UserPage(props: { user: User }) {
|
||||||
{showConfetti && (
|
{showConfetti && (
|
||||||
<FullscreenConfetti recycle={false} numberOfPieces={300} />
|
<FullscreenConfetti recycle={false} numberOfPieces={300} />
|
||||||
)}
|
)}
|
||||||
<BettingStreakModal
|
<Col className="relative">
|
||||||
isOpen={showBettingStreakModal}
|
<Row className="relative px-4 pt-4">
|
||||||
setOpen={setShowBettingStreakModal}
|
|
||||||
currentUser={currentUser}
|
|
||||||
/>
|
|
||||||
{showLoansModal && (
|
|
||||||
<LoansModal isOpen={showLoansModal} setOpen={setShowLoansModal} />
|
|
||||||
)}
|
|
||||||
{/* Banner image up top, with an circle avatar overlaid */}
|
|
||||||
<div
|
|
||||||
className="h-32 w-full bg-cover bg-center sm:h-40"
|
|
||||||
style={{
|
|
||||||
backgroundImage: `url(${bannerUrl})`,
|
|
||||||
}}
|
|
||||||
></div>
|
|
||||||
<div className="relative mb-20">
|
|
||||||
<div className="absolute -top-10 left-4">
|
|
||||||
<Avatar
|
<Avatar
|
||||||
username={user.username}
|
username={user.username}
|
||||||
avatarUrl={user.avatarUrl}
|
avatarUrl={user.avatarUrl}
|
||||||
size={24}
|
size={24}
|
||||||
className="bg-white ring-4 ring-white"
|
className="bg-white shadow-sm shadow-indigo-300"
|
||||||
/>
|
/>
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Top right buttons (e.g. edit, follow) */}
|
|
||||||
<div className="absolute right-0 top-0 mt-2 mr-4">
|
|
||||||
{!isCurrentUser && <UserFollowButton userId={user.id} />}
|
|
||||||
{isCurrentUser && (
|
{isCurrentUser && (
|
||||||
<SiteLink className="btn-sm btn" href="/profile">
|
<div className="absolute z-50 ml-16 mt-16 rounded-full bg-indigo-600 p-2 text-white shadow-sm shadow-indigo-300">
|
||||||
<PencilIcon className="h-5 w-5" />{' '}
|
<SiteLink href="/profile">
|
||||||
<div className="ml-2">Edit</div>
|
<PencilIcon className="h-5" />{' '}
|
||||||
</SiteLink>
|
</SiteLink>
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Profile details: name, username, bio, and link to twitter/discord */}
|
<Col className="w-full gap-4 pl-5">
|
||||||
<Col className="mx-4 -mt-6">
|
<div className="flex flex-col gap-2 sm:flex-row sm:justify-between">
|
||||||
<Row className={'flex-wrap justify-between gap-y-2'}>
|
|
||||||
<Col>
|
<Col>
|
||||||
<span className="break-anywhere text-2xl font-bold">
|
<span className="break-anywhere text-lg font-bold sm:text-2xl">
|
||||||
{user.name}
|
{user.name}
|
||||||
</span>
|
</span>
|
||||||
<span className="text-gray-500">@{user.username}</span>
|
<span className="sm:text-md text-greyscale-4 text-sm">
|
||||||
</Col>
|
@{user.username}
|
||||||
<Col className={'justify-center'}>
|
|
||||||
<Row className={'gap-3'}>
|
|
||||||
<Col className={'items-center text-gray-500'}>
|
|
||||||
<span
|
|
||||||
className={clsx(
|
|
||||||
'text-md',
|
|
||||||
profit >= 0 ? 'text-green-600' : 'text-red-400'
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{formatMoney(profit)}
|
|
||||||
</span>
|
</span>
|
||||||
<span>profit</span>
|
|
||||||
</Col>
|
</Col>
|
||||||
<Col
|
{isCurrentUser && (
|
||||||
className={clsx(
|
<ProfilePrivateStats
|
||||||
'cursor-pointer items-center text-gray-500',
|
currentUser={currentUser}
|
||||||
isCurrentUser && !hasCompletedStreakToday(user)
|
profit={profit}
|
||||||
? 'grayscale'
|
user={user}
|
||||||
: 'grayscale-0'
|
router={router}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
onClick={() => setShowBettingStreakModal(true)}
|
{!isCurrentUser && <UserFollowButton userId={user.id} />}
|
||||||
>
|
</div>
|
||||||
<span>🔥 {user.currentBettingStreak ?? 0}</span>
|
<ProfilePublicStats
|
||||||
<span>streak</span>
|
className="sm:text-md text-greyscale-6 hidden text-sm md:inline"
|
||||||
</Col>
|
user={user}
|
||||||
<Col
|
/>
|
||||||
className={
|
|
||||||
'flex-shrink-0 cursor-pointer items-center text-gray-500'
|
|
||||||
}
|
|
||||||
onClick={() => setShowLoansModal(true)}
|
|
||||||
>
|
|
||||||
<span className="text-green-600">
|
|
||||||
🏦 {formatMoney(user.nextLoanCached ?? 0)}
|
|
||||||
</span>
|
|
||||||
<span>next loan</span>
|
|
||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
</Col>
|
<Col className="mx-4 mt-2">
|
||||||
</Row>
|
<Spacer h={1} />
|
||||||
<Spacer h={4} />
|
<ProfilePublicStats
|
||||||
|
className="text-greyscale-6 text-sm md:hidden"
|
||||||
|
user={user}
|
||||||
|
/>
|
||||||
|
<Spacer h={1} />
|
||||||
{user.bio && (
|
{user.bio && (
|
||||||
<>
|
<>
|
||||||
<div>
|
<div className="sm:text-md mt-2 text-sm sm:mt-0">
|
||||||
<Linkify text={user.bio}></Linkify>
|
<Linkify text={user.bio}></Linkify>
|
||||||
</div>
|
</div>
|
||||||
<Spacer h={4} />
|
<Spacer h={2} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{(user.website || user.twitterHandle || user.discordHandle) && (
|
{(user.website || user.twitterHandle || user.discordHandle) && (
|
||||||
<Row className="mb-5 flex-wrap items-center gap-2 sm:gap-4">
|
<Row className="mb-2 flex-wrap items-center gap-2 sm:gap-4">
|
||||||
{user.website && (
|
{user.website && (
|
||||||
<SiteLink
|
<SiteLink
|
||||||
href={
|
href={
|
||||||
|
@ -190,7 +142,9 @@ export function UserPage(props: { user: User }) {
|
||||||
>
|
>
|
||||||
<Row className="items-center gap-1">
|
<Row className="items-center gap-1">
|
||||||
<LinkIcon className="h-4 w-4" />
|
<LinkIcon className="h-4 w-4" />
|
||||||
<span className="text-sm text-gray-500">{user.website}</span>
|
<span className="text-greyscale-4 text-sm">
|
||||||
|
{user.website}
|
||||||
|
</span>
|
||||||
</Row>
|
</Row>
|
||||||
</SiteLink>
|
</SiteLink>
|
||||||
)}
|
)}
|
||||||
|
@ -209,7 +163,7 @@ export function UserPage(props: { user: User }) {
|
||||||
className="h-4 w-4"
|
className="h-4 w-4"
|
||||||
alt="Twitter"
|
alt="Twitter"
|
||||||
/>
|
/>
|
||||||
<span className="text-sm text-gray-500">
|
<span className="text-greyscale-4 text-sm">
|
||||||
{user.twitterHandle}
|
{user.twitterHandle}
|
||||||
</span>
|
</span>
|
||||||
</Row>
|
</Row>
|
||||||
|
@ -224,7 +178,7 @@ export function UserPage(props: { user: User }) {
|
||||||
className="h-4 w-4"
|
className="h-4 w-4"
|
||||||
alt="Discord"
|
alt="Discord"
|
||||||
/>
|
/>
|
||||||
<span className="text-sm text-gray-500">
|
<span className="text-greyscale-4 text-sm">
|
||||||
{user.discordHandle}
|
{user.discordHandle}
|
||||||
</span>
|
</span>
|
||||||
</Row>
|
</Row>
|
||||||
|
@ -232,72 +186,48 @@ export function UserPage(props: { user: User }) {
|
||||||
)}
|
)}
|
||||||
</Row>
|
</Row>
|
||||||
)}
|
)}
|
||||||
{currentUser?.id === user.id && REFERRAL_AMOUNT > 0 && (
|
|
||||||
<Row
|
|
||||||
className={
|
|
||||||
'mb-5 w-full items-center justify-center gap-2 rounded-md border-2 border-indigo-100 bg-indigo-50 p-2 text-indigo-600'
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
<SiteLink href="/referrals">
|
|
||||||
Earn {formatMoney(REFERRAL_AMOUNT)} when you refer a friend!
|
|
||||||
</SiteLink>{' '}
|
|
||||||
You've gotten{' '}
|
|
||||||
<ReferralsButton user={user} currentUser={currentUser} />
|
|
||||||
</span>
|
|
||||||
<ShareIconButton
|
|
||||||
copyPayload={`https://${ENV_CONFIG.domain}?referrer=${currentUser.username}`}
|
|
||||||
toastClassName={'sm:-left-40 -left-40 min-w-[250%]'}
|
|
||||||
buttonClassName={'h-10 w-10'}
|
|
||||||
iconClassName={'h-8 w-8 text-indigo-700'}
|
|
||||||
/>
|
|
||||||
</Row>
|
|
||||||
)}
|
|
||||||
<QueryUncontrolledTabs
|
<QueryUncontrolledTabs
|
||||||
className="mb-4"
|
|
||||||
currentPageForAnalytics={'profile'}
|
currentPageForAnalytics={'profile'}
|
||||||
labelClassName={'pb-2 pt-1 '}
|
labelClassName={'pb-2 pt-1 sm:pt-4 '}
|
||||||
tabs={[
|
tabs={[
|
||||||
{
|
{
|
||||||
title: 'Markets',
|
title: 'Portfolio',
|
||||||
content: (
|
tabIcon: <FolderIcon className="h-5" />,
|
||||||
<CreatorContractsList user={currentUser} creator={user} />
|
|
||||||
),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: 'Comments',
|
|
||||||
content: (
|
|
||||||
<Col>
|
|
||||||
<UserCommentsList user={user} />
|
|
||||||
</Col>
|
|
||||||
),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
title: capitalize(PAST_BETS),
|
|
||||||
content: (
|
content: (
|
||||||
<>
|
<>
|
||||||
|
<Spacer h={4} />
|
||||||
|
<PortfolioValueSection userId={user.id} />
|
||||||
|
<Spacer h={4} />
|
||||||
<BetsList user={user} />
|
<BetsList user={user} />
|
||||||
</>
|
</>
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: 'Stats',
|
title: 'Markets',
|
||||||
|
tabIcon: <ScaleIcon className="h-5" />,
|
||||||
content: (
|
content: (
|
||||||
<Col className="mb-8">
|
<>
|
||||||
<Row className="mb-8 flex-wrap items-center gap-x-6 gap-y-2">
|
<Spacer h={4} />
|
||||||
<FollowingButton user={user} />
|
<CreatorContractsList user={currentUser} creator={user} />
|
||||||
<FollowersButton user={user} />
|
</>
|
||||||
<ReferralsButton user={user} />
|
),
|
||||||
<GroupsButton user={user} />
|
},
|
||||||
<UserLikesButton user={user} />
|
{
|
||||||
</Row>
|
title: 'Comments',
|
||||||
<PortfolioValueSection userId={user.id} />
|
tabIcon: <ChatIcon className="h-5" />,
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
<Spacer h={4} />
|
||||||
|
<Col>
|
||||||
|
<UserCommentsList user={user} />
|
||||||
</Col>
|
</Col>
|
||||||
|
</>
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
]}
|
]}
|
||||||
/>
|
/>
|
||||||
</Col>
|
</Col>
|
||||||
|
</Col>
|
||||||
</Page>
|
</Page>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -314,3 +244,88 @@ export function defaultBannerUrl(userId: string) {
|
||||||
]
|
]
|
||||||
return defaultBanner[genHash(userId)() % defaultBanner.length]
|
return defaultBanner[genHash(userId)() % defaultBanner.length]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function ProfilePrivateStats(props: {
|
||||||
|
currentUser: User | null | undefined
|
||||||
|
profit: number
|
||||||
|
user: User
|
||||||
|
router: NextRouter
|
||||||
|
}) {
|
||||||
|
const { currentUser, profit, user, router } = props
|
||||||
|
const [showBettingStreakModal, setShowBettingStreakModal] = useState(false)
|
||||||
|
const [showLoansModal, setShowLoansModal] = useState(false)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const showBettingStreak = router.query['show'] === 'betting-streak'
|
||||||
|
setShowBettingStreakModal(showBettingStreak)
|
||||||
|
|
||||||
|
const showLoansModel = router.query['show'] === 'loans'
|
||||||
|
setShowLoansModal(showLoansModel)
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [])
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Row className={'justify-between gap-4 sm:justify-end'}>
|
||||||
|
<Col className={'text-greyscale-4 text-md sm:text-lg'}>
|
||||||
|
<span
|
||||||
|
className={clsx(profit >= 0 ? 'text-green-600' : 'text-red-400')}
|
||||||
|
>
|
||||||
|
{formatMoney(profit)}
|
||||||
|
</span>
|
||||||
|
<span className="mx-auto text-xs sm:text-sm">profit</span>
|
||||||
|
</Col>
|
||||||
|
<Col
|
||||||
|
className={clsx('text-,d cursor-pointer sm:text-lg ')}
|
||||||
|
onClick={() => setShowBettingStreakModal(true)}
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
!hasCompletedStreakToday(user)
|
||||||
|
? 'opacity-50 grayscale'
|
||||||
|
: 'grayscale-0'
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
🔥 {user.currentBettingStreak ?? 0}
|
||||||
|
</span>
|
||||||
|
<span className="text-greyscale-4 mx-auto text-xs sm:text-sm">
|
||||||
|
streak
|
||||||
|
</span>
|
||||||
|
</Col>
|
||||||
|
<Col
|
||||||
|
className={
|
||||||
|
'text-greyscale-4 text-md flex-shrink-0 cursor-pointer sm:text-lg'
|
||||||
|
}
|
||||||
|
onClick={() => setShowLoansModal(true)}
|
||||||
|
>
|
||||||
|
<span className="text-green-600">
|
||||||
|
🏦 {formatMoney(user.nextLoanCached ?? 0)}
|
||||||
|
</span>
|
||||||
|
<span className="mx-auto text-xs sm:text-sm">next loan</span>
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
|
{BettingStreakModal && (
|
||||||
|
<BettingStreakModal
|
||||||
|
isOpen={showBettingStreakModal}
|
||||||
|
setOpen={setShowBettingStreakModal}
|
||||||
|
currentUser={currentUser}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{showLoansModal && (
|
||||||
|
<LoansModal isOpen={showLoansModal} setOpen={setShowLoansModal} />
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProfilePublicStats(props: { user: User; className?: string }) {
|
||||||
|
const { user, className } = props
|
||||||
|
return (
|
||||||
|
<Row className={'flex-wrap items-center gap-3'}>
|
||||||
|
<FollowingButton user={user} className={className} />
|
||||||
|
<FollowersButton user={user} className={className} />
|
||||||
|
{/* <ReferralsButton user={user} className={className} /> */}
|
||||||
|
<GroupsButton user={user} className={className} />
|
||||||
|
{/* <UserLikesButton user={user} className={className} /> */}
|
||||||
|
</Row>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -13,7 +13,6 @@ import { Page } from 'web/components/page'
|
||||||
import { SEO } from 'web/components/SEO'
|
import { SEO } from 'web/components/SEO'
|
||||||
import { SiteLink } from 'web/components/site-link'
|
import { SiteLink } from 'web/components/site-link'
|
||||||
import { Title } from 'web/components/title'
|
import { Title } from 'web/components/title'
|
||||||
import { defaultBannerUrl } from 'web/components/user-page'
|
|
||||||
import { generateNewApiKey } from 'web/lib/api/api-key'
|
import { generateNewApiKey } from 'web/lib/api/api-key'
|
||||||
import { changeUserInfo } from 'web/lib/firebase/api'
|
import { changeUserInfo } from 'web/lib/firebase/api'
|
||||||
import { redirectIfLoggedOut } from 'web/lib/firebase/server-auth'
|
import { redirectIfLoggedOut } from 'web/lib/firebase/server-auth'
|
||||||
|
@ -176,27 +175,6 @@ export default function ProfilePage(props: {
|
||||||
onBlur={updateUsername}
|
onBlur={updateUsername}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* TODO: Allow users with M$ 2000 of assets to set custom banners */}
|
|
||||||
{/* <EditUserField
|
|
||||||
user={user}
|
|
||||||
field="bannerUrl"
|
|
||||||
label="Banner Url"
|
|
||||||
isEditing={isEditing}
|
|
||||||
/> */}
|
|
||||||
<label className="label">
|
|
||||||
Banner image{' '}
|
|
||||||
<span className="text-sm text-gray-400">Not editable for now</span>
|
|
||||||
</label>
|
|
||||||
<div
|
|
||||||
className="h-32 w-full bg-cover bg-center sm:h-40"
|
|
||||||
style={{
|
|
||||||
backgroundImage: `url(${
|
|
||||||
user.bannerUrl || defaultBannerUrl(user.id)
|
|
||||||
})`,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
|
|
||||||
{(
|
{(
|
||||||
[
|
[
|
||||||
['bio', 'Bio'],
|
['bio', 'Bio'],
|
||||||
|
|
Loading…
Reference in New Issue
Block a user