2023-07-23 10:08:43 -05:00
|
|
|
import { gql } from 'graphql-tag'
|
2022-12-19 16:27:52 -06:00
|
|
|
import Link from 'next/link'
|
|
|
|
import { useRouter } from 'next/router'
|
2024-03-20 01:37:31 +01:00
|
|
|
import { getGetServerSideProps } from '@/api/ssrApollo'
|
|
|
|
import { CopyInput, Select, DatePicker } from '@/components/form'
|
|
|
|
import { CenterLayout } from '@/components/layout'
|
|
|
|
import { useMe } from '@/components/me'
|
2023-07-23 10:08:43 -05:00
|
|
|
import { useQuery } from '@apollo/client'
|
2024-03-20 01:37:31 +01:00
|
|
|
import PageLoading from '@/components/page-loading'
|
|
|
|
import { WHENS } from '@/lib/constants'
|
2023-07-24 17:50:12 -05:00
|
|
|
import dynamic from 'next/dynamic'
|
2024-03-20 01:37:31 +01:00
|
|
|
import { numWithUnits } from '@/lib/format'
|
|
|
|
import { whenToFrom } from '@/lib/time'
|
2024-03-22 23:47:21 -04:00
|
|
|
import { WhenComposedChartSkeleton } from '@/components/charts-skeletons'
|
2023-07-24 17:50:12 -05:00
|
|
|
|
2024-03-20 01:37:31 +01:00
|
|
|
const WhenComposedChart = dynamic(() => import('@/components/charts').then(mod => mod.WhenComposedChart), {
|
2024-03-22 23:47:21 -04:00
|
|
|
loading: () => <WhenComposedChartSkeleton />
|
2023-07-24 17:50:12 -05:00
|
|
|
})
|
2022-12-19 16:27:52 -06:00
|
|
|
|
2023-07-23 10:08:43 -05:00
|
|
|
const REFERRALS = gql`
|
2023-11-08 21:15:36 -03:00
|
|
|
query Referrals($when: String!, $from: String, $to: String)
|
2023-07-23 10:08:43 -05:00
|
|
|
{
|
2023-11-08 21:15:36 -03:00
|
|
|
referrals(when: $when, from: $from, to: $to) {
|
2023-07-23 10:08:43 -05:00
|
|
|
totalSats
|
|
|
|
totalReferrals
|
|
|
|
stats {
|
|
|
|
time
|
|
|
|
data {
|
|
|
|
name
|
|
|
|
value
|
2022-12-19 16:27:52 -06:00
|
|
|
}
|
|
|
|
}
|
2023-07-23 10:08:43 -05:00
|
|
|
}
|
|
|
|
}`
|
2022-12-19 16:27:52 -06:00
|
|
|
|
2023-08-28 12:52:15 -05:00
|
|
|
export const getServerSideProps = getGetServerSideProps({ query: REFERRALS, authRequired: true })
|
2023-07-23 10:08:43 -05:00
|
|
|
|
|
|
|
export default function Referrals ({ ssrData }) {
|
2022-12-19 16:27:52 -06:00
|
|
|
const router = useRouter()
|
|
|
|
const me = useMe()
|
2023-07-23 10:08:43 -05:00
|
|
|
|
2023-11-08 21:15:36 -03:00
|
|
|
const select = async values => {
|
|
|
|
const { when, ...query } = values
|
|
|
|
|
|
|
|
if (when !== 'custom') { delete query.from; delete query.to }
|
|
|
|
if (query.from && !query.to) return
|
|
|
|
|
|
|
|
await router.push({
|
|
|
|
pathname: `/referrals/${when}`,
|
|
|
|
query
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const { data } = useQuery(REFERRALS, { variables: { when: router.query.when, from: router.query.from, to: router.query.to } })
|
2023-07-23 10:08:43 -05:00
|
|
|
if (!data && !ssrData) return <PageLoading />
|
|
|
|
|
2023-07-24 13:35:05 -05:00
|
|
|
const { referrals: { totalSats, totalReferrals, stats } } = data || ssrData
|
2023-07-23 10:08:43 -05:00
|
|
|
|
2023-11-08 21:15:36 -03:00
|
|
|
const when = router.query.when
|
|
|
|
|
2022-12-19 16:27:52 -06:00
|
|
|
return (
|
2023-07-23 10:08:43 -05:00
|
|
|
<CenterLayout footerLinks>
|
2023-11-08 21:15:36 -03:00
|
|
|
<div className='fw-bold text-muted text-center pt-5 pb-3 d-flex align-items-center justify-content-center flex-wrap'>
|
|
|
|
<h4 className='fw-bold text-muted text-center d-flex align-items-center justify-content-center'>
|
|
|
|
{numWithUnits(totalReferrals, { unitPlural: 'referrals', unitSingular: 'referral' })} & {numWithUnits(totalSats, { abbreviate: false })} in the last
|
|
|
|
<Select
|
|
|
|
groupClassName='mb-0 mx-2'
|
|
|
|
className='w-auto'
|
|
|
|
name='when'
|
|
|
|
size='sm'
|
|
|
|
items={WHENS}
|
|
|
|
value={router.query.when || 'day'}
|
|
|
|
noForm
|
|
|
|
onChange={(formik, e) => {
|
2023-11-14 10:23:44 -06:00
|
|
|
const range = e.target.value === 'custom' ? { from: whenToFrom(when), to: Date.now() } : {}
|
2023-11-08 21:15:36 -03:00
|
|
|
select({ when: e.target.value, ...range })
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</h4>
|
|
|
|
{when === 'custom' &&
|
|
|
|
<DatePicker
|
|
|
|
noForm
|
|
|
|
fromName='from'
|
|
|
|
toName='to'
|
|
|
|
className='p-0 px-2 mb-2'
|
|
|
|
onChange={(formik, [from, to], e) => {
|
2023-11-14 10:23:44 -06:00
|
|
|
select({ when, from: from.getTime(), to: to.getTime() })
|
2023-11-08 21:15:36 -03:00
|
|
|
}}
|
|
|
|
from={router.query.from}
|
|
|
|
to={router.query.to}
|
|
|
|
when={router.query.when}
|
|
|
|
/>}
|
|
|
|
</div>
|
2023-08-14 15:36:54 -05:00
|
|
|
<WhenComposedChart data={stats} lineNames={['sats']} barNames={['referrals']} barAxis='right' />
|
2022-12-19 16:27:52 -06:00
|
|
|
|
|
|
|
<div
|
|
|
|
className='text-small pt-5 px-3 d-flex w-100 align-items-center'
|
|
|
|
>
|
2023-07-24 13:35:05 -05:00
|
|
|
<div className='nav-item text-muted pe-2' style={{ 'white-space': 'nowrap' }}>referral link:</div>
|
2022-12-19 16:27:52 -06:00
|
|
|
<CopyInput
|
|
|
|
size='sm'
|
|
|
|
groupClassName='mb-0 w-100'
|
|
|
|
readOnly
|
|
|
|
noForm
|
|
|
|
placeholder={`https://stacker.news/r/${me.name}`}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<ul className='py-3 text-muted'>
|
|
|
|
<li>{`appending /r/${me.name} to any SN link makes it a ref link`}
|
|
|
|
<ul>
|
|
|
|
<li>e.g. https://stacker.news/items/1/r/{me.name}</li>
|
|
|
|
</ul>
|
|
|
|
</li>
|
|
|
|
<li>earn 21% of boost and job fees spent by referred stackers</li>
|
2023-06-19 13:21:55 -05:00
|
|
|
<li>earn 2.1% of all zaps received by referred stackers</li>
|
2023-07-23 10:08:43 -05:00
|
|
|
<li><Link href='/invites'>invite links</Link> are also implicitly referral links</li>
|
2022-12-19 16:27:52 -06:00
|
|
|
</ul>
|
2023-07-23 10:08:43 -05:00
|
|
|
</CenterLayout>
|
2022-12-19 16:27:52 -06:00
|
|
|
)
|
|
|
|
}
|