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:
ingawei 2022-09-26 18:01:13 -05:00 committed by GitHub
parent d612192109
commit 2fe9fe593d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 529 additions and 337 deletions

View File

@ -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 ? (

View File

@ -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}

View File

@ -128,6 +128,7 @@ export function CreatorContractsList(props: {
creatorId: creator.id, creatorId: creator.id,
}} }}
persistPrefix={`user-${creator.id}`} persistPrefix={`user-${creator.id}`}
profile={true}
/> />
) )
} }

View File

@ -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>

View File

@ -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>

View File

@ -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>

View File

@ -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
}

View File

@ -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>
)
}

View File

@ -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>

View File

@ -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>

View File

@ -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>
)
}

View File

@ -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'],