2021-04-22 17:14:32 -05:00
|
|
|
|
import { LightningConsumer } from './lightning'
|
2021-12-05 11:37:55 -06:00
|
|
|
|
import UpBolt from '../svgs/bolt.svg'
|
2021-04-22 17:14:32 -05:00
|
|
|
|
import styles from './upvote.module.css'
|
2021-04-26 16:55:15 -05:00
|
|
|
|
import { gql, useMutation } from '@apollo/client'
|
2021-09-24 18:04:59 -05:00
|
|
|
|
import { signIn } from 'next-auth/client'
|
2021-05-20 16:32:59 -05:00
|
|
|
|
import { useFundError } from './fund-error'
|
2021-07-08 13:42:57 -05:00
|
|
|
|
import ActionTooltip from './action-tooltip'
|
2021-09-10 13:55:36 -05:00
|
|
|
|
import { useItemAct } from './item-act'
|
2021-09-12 11:55:38 -05:00
|
|
|
|
import { useMe } from './me'
|
2021-12-13 13:49:34 -06:00
|
|
|
|
import Rainbow from '../lib/rainbow'
|
2021-12-09 14:40:40 -06:00
|
|
|
|
import { useRef, useState } from 'react'
|
2021-10-30 11:52:24 -05:00
|
|
|
|
import LongPressable from 'react-longpressable'
|
2021-12-09 14:40:40 -06:00
|
|
|
|
import { Overlay, Popover } from 'react-bootstrap'
|
|
|
|
|
|
|
|
|
|
const getColor = (meSats) => {
|
|
|
|
|
if (!meSats || meSats <= 10) {
|
|
|
|
|
return 'var(--secondary)'
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const idx = Math.min(
|
2021-12-13 13:49:34 -06:00
|
|
|
|
Math.floor((Math.log(meSats) / Math.log(10000)) * (Rainbow.length - 1)),
|
2021-12-09 14:40:40 -06:00
|
|
|
|
Rainbow.length - 1)
|
|
|
|
|
return Rainbow[idx]
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-12 16:09:12 -05:00
|
|
|
|
const UpvotePopover = ({ target, show, handleClose }) => {
|
|
|
|
|
const me = useMe()
|
|
|
|
|
return (
|
|
|
|
|
<Overlay
|
|
|
|
|
show={show}
|
|
|
|
|
target={target}
|
|
|
|
|
placement='right'
|
|
|
|
|
>
|
|
|
|
|
<Popover id='popover-basic'>
|
|
|
|
|
<Popover.Title className='d-flex justify-content-between alert-dismissible' as='h3'>Tipping
|
|
|
|
|
<button type='button' className='close' onClick={handleClose}><span aria-hidden='true'>×</span><span className='sr-only'>Close alert</span></button>
|
|
|
|
|
</Popover.Title>
|
|
|
|
|
<Popover.Content>
|
|
|
|
|
<div className='mb-2'>Press the bolt again to tip {me?.tipDefault || 1} more sat{me?.tipDefault > 1 ? 's' : ''}.</div>
|
|
|
|
|
<div>Repeatedly press the bolt to tip more sats.</div>
|
|
|
|
|
</Popover.Content>
|
|
|
|
|
</Popover>
|
|
|
|
|
</Overlay>
|
|
|
|
|
)
|
|
|
|
|
}
|
2021-12-09 14:40:40 -06:00
|
|
|
|
|
|
|
|
|
const TipPopover = ({ target, show, handleClose }) => (
|
|
|
|
|
<Overlay
|
|
|
|
|
show={show}
|
|
|
|
|
target={target}
|
|
|
|
|
placement='right'
|
|
|
|
|
>
|
|
|
|
|
<Popover id='popover-basic'>
|
|
|
|
|
<Popover.Title className='d-flex justify-content-between alert-dismissible' as='h3'>Press and hold
|
|
|
|
|
<button type='button' class='close' onClick={handleClose}><span aria-hidden='true'>×</span><span class='sr-only'>Close alert</span></button>
|
|
|
|
|
</Popover.Title>
|
|
|
|
|
<Popover.Content>
|
2021-12-13 13:49:34 -06:00
|
|
|
|
<div className='mb-2'>Press and hold bolt to tip a custom amount.</div>
|
|
|
|
|
<div>As you tip more, the bolt color follows the rainbow.</div>
|
2021-12-09 14:40:40 -06:00
|
|
|
|
</Popover.Content>
|
|
|
|
|
</Popover>
|
|
|
|
|
</Overlay>
|
|
|
|
|
)
|
2021-04-26 16:55:15 -05:00
|
|
|
|
|
2021-09-10 16:13:52 -05:00
|
|
|
|
export default function UpVote ({ item, className }) {
|
2021-05-20 16:32:59 -05:00
|
|
|
|
const { setError } = useFundError()
|
2021-09-10 13:55:36 -05:00
|
|
|
|
const { setItem } = useItemAct()
|
2021-12-09 14:40:40 -06:00
|
|
|
|
const [voteShow, _setVoteShow] = useState(false)
|
|
|
|
|
const [tipShow, _setTipShow] = useState(false)
|
|
|
|
|
const ref = useRef()
|
2021-09-12 11:55:38 -05:00
|
|
|
|
const me = useMe()
|
2021-12-09 14:40:40 -06:00
|
|
|
|
const [setWalkthrough] = useMutation(
|
|
|
|
|
gql`
|
|
|
|
|
mutation setWalkthrough($upvotePopover: Boolean, $tipPopover: Boolean) {
|
|
|
|
|
setWalkthrough(upvotePopover: $upvotePopover, tipPopover: $tipPopover)
|
|
|
|
|
}`
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const setVoteShow = (yes) => {
|
|
|
|
|
if (!me) return
|
|
|
|
|
|
2022-03-10 14:09:09 -06:00
|
|
|
|
// if they haven't seen the walkthrough and they have sats
|
|
|
|
|
if (yes && !me.upvotePopover && me.sats) {
|
|
|
|
|
_setVoteShow(true)
|
2021-12-09 14:40:40 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (voteShow && !yes) {
|
2022-03-10 14:09:09 -06:00
|
|
|
|
_setVoteShow(false)
|
2021-12-09 14:40:40 -06:00
|
|
|
|
setWalkthrough({ variables: { upvotePopover: true } })
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const setTipShow = (yes) => {
|
|
|
|
|
if (!me) return
|
|
|
|
|
|
|
|
|
|
// if we want to show it, yet we still haven't shown
|
2022-03-10 14:09:09 -06:00
|
|
|
|
if (yes && !me.tipPopover && me.sats) {
|
|
|
|
|
_setTipShow(true)
|
2021-12-09 14:40:40 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// if it's currently showing and we want to hide it
|
|
|
|
|
if (tipShow && !yes) {
|
2022-03-10 14:09:09 -06:00
|
|
|
|
_setTipShow(false)
|
2021-12-09 14:40:40 -06:00
|
|
|
|
setWalkthrough({ variables: { tipPopover: true } })
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-08 16:51:23 -05:00
|
|
|
|
const [act] = useMutation(
|
2021-04-26 16:55:15 -05:00
|
|
|
|
gql`
|
2022-01-20 17:04:12 -06:00
|
|
|
|
mutation act($id: ID!, $sats: Int!) {
|
|
|
|
|
act(id: $id, sats: $sats) {
|
|
|
|
|
vote,
|
2021-09-10 16:13:52 -05:00
|
|
|
|
sats
|
|
|
|
|
}
|
2021-04-26 16:55:15 -05:00
|
|
|
|
}`, {
|
2022-01-20 17:04:12 -06:00
|
|
|
|
update (cache, { data: { act: { vote, sats } } }) {
|
2021-04-26 16:55:15 -05:00
|
|
|
|
cache.modify({
|
2021-09-10 16:13:52 -05:00
|
|
|
|
id: `Item:${item.id}`,
|
2021-04-26 16:55:15 -05:00
|
|
|
|
fields: {
|
|
|
|
|
sats (existingSats = 0) {
|
2022-01-20 17:04:12 -06:00
|
|
|
|
return existingSats + sats
|
2021-04-27 16:30:58 -05:00
|
|
|
|
},
|
2021-12-05 11:37:55 -06:00
|
|
|
|
meSats (existingSats = 0) {
|
2022-01-20 17:04:12 -06:00
|
|
|
|
if (existingSats === 0) {
|
|
|
|
|
setVoteShow(true)
|
|
|
|
|
} else {
|
|
|
|
|
setTipShow(true)
|
2021-09-10 16:13:52 -05:00
|
|
|
|
}
|
2022-01-20 17:04:12 -06:00
|
|
|
|
return existingSats + sats
|
2021-09-10 16:13:52 -05:00
|
|
|
|
},
|
2022-01-20 17:04:12 -06:00
|
|
|
|
upvotes (existingUpvotes = 0) {
|
|
|
|
|
return existingUpvotes + vote
|
2021-04-26 16:55:15 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
)
|
2021-04-22 17:14:32 -05:00
|
|
|
|
|
2021-09-12 11:55:38 -05:00
|
|
|
|
const overlayText = () => {
|
2022-01-20 17:04:12 -06:00
|
|
|
|
if (me?.tipDefault) {
|
|
|
|
|
return `${me.tipDefault} sat${me.tipDefault > 1 ? 's' : ''}`
|
2021-09-12 11:55:38 -05:00
|
|
|
|
}
|
2022-01-20 17:04:12 -06:00
|
|
|
|
return '1 sat'
|
2021-09-12 11:55:38 -05:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-05 11:37:55 -06:00
|
|
|
|
const color = getColor(item?.meSats)
|
2021-04-22 17:14:32 -05:00
|
|
|
|
return (
|
|
|
|
|
<LightningConsumer>
|
|
|
|
|
{({ strike }) =>
|
2022-01-20 13:03:48 -06:00
|
|
|
|
<div ref={ref} className='upvoteParent'>
|
2021-12-09 14:40:40 -06:00
|
|
|
|
<LongPressable
|
|
|
|
|
onLongPress={
|
2021-10-30 11:52:24 -05:00
|
|
|
|
async (e) => {
|
2022-01-20 17:23:21 -06:00
|
|
|
|
if (!item) return
|
2021-10-30 11:52:24 -05:00
|
|
|
|
|
|
|
|
|
// we can't tip ourselves
|
2022-01-20 17:04:12 -06:00
|
|
|
|
if (item?.mine) {
|
2021-10-30 11:52:24 -05:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-09 14:40:40 -06:00
|
|
|
|
setTipShow(false)
|
2021-10-30 11:52:24 -05:00
|
|
|
|
setItem({ itemId: item.id, act, strike })
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-09 14:40:40 -06:00
|
|
|
|
onShortPress={
|
2021-09-24 18:04:59 -05:00
|
|
|
|
me
|
2021-06-24 18:56:01 -05:00
|
|
|
|
? async (e) => {
|
2022-01-20 17:23:21 -06:00
|
|
|
|
if (!item) return
|
2021-09-12 11:55:38 -05:00
|
|
|
|
|
|
|
|
|
// we can't tip ourselves
|
2022-01-20 17:04:12 -06:00
|
|
|
|
if (item?.mine) {
|
2021-09-12 11:55:38 -05:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 17:04:12 -06:00
|
|
|
|
if (item?.meSats) {
|
2021-12-09 14:40:40 -06:00
|
|
|
|
setVoteShow(false)
|
2021-09-10 13:55:36 -05:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-24 18:56:01 -05:00
|
|
|
|
strike()
|
2021-09-10 13:55:36 -05:00
|
|
|
|
|
2021-05-20 16:32:59 -05:00
|
|
|
|
try {
|
2022-01-20 17:23:21 -06:00
|
|
|
|
await act({
|
|
|
|
|
variables: { id: item.id, sats: me.tipDefault || 1 },
|
|
|
|
|
optimisticResponse: {
|
|
|
|
|
act: {
|
|
|
|
|
id: `Item:${item.id}`,
|
|
|
|
|
sats: me.tipDefault || 1,
|
|
|
|
|
vote: 0
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
2021-05-20 16:32:59 -05:00
|
|
|
|
} catch (error) {
|
|
|
|
|
if (error.toString().includes('insufficient funds')) {
|
|
|
|
|
setError(true)
|
|
|
|
|
return
|
|
|
|
|
}
|
2021-05-20 14:41:21 -05:00
|
|
|
|
throw new Error({ message: error.toString() })
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
: signIn
|
|
|
|
|
}
|
2021-12-09 14:40:40 -06:00
|
|
|
|
>
|
2022-01-20 17:04:12 -06:00
|
|
|
|
<ActionTooltip notForm disable={item?.mine} overlayText={overlayText()}>
|
2021-12-09 14:40:40 -06:00
|
|
|
|
<div
|
2022-01-20 17:04:12 -06:00
|
|
|
|
className={`${item?.mine ? styles.noSelfTips : ''}
|
2021-12-09 14:40:40 -06:00
|
|
|
|
${styles.upvoteWrapper}`}
|
|
|
|
|
>
|
2021-12-06 09:32:33 -06:00
|
|
|
|
<UpBolt
|
|
|
|
|
width={24}
|
|
|
|
|
height={24}
|
|
|
|
|
className={
|
2021-12-09 14:40:40 -06:00
|
|
|
|
`${styles.upvote}
|
|
|
|
|
${className || ''}
|
2022-01-20 17:04:12 -06:00
|
|
|
|
${item?.mine ? styles.noSelfTips : ''}
|
2022-01-19 17:00:20 -06:00
|
|
|
|
${item?.meSats ? styles.voted : ''}`
|
2021-12-09 14:40:40 -06:00
|
|
|
|
}
|
2022-01-19 17:00:20 -06:00
|
|
|
|
style={item?.meSats
|
2021-12-06 09:32:33 -06:00
|
|
|
|
? {
|
|
|
|
|
fill: color,
|
|
|
|
|
filter: `drop-shadow(0 0 6px ${color}90)`
|
|
|
|
|
}
|
|
|
|
|
: undefined}
|
|
|
|
|
/>
|
2021-12-09 14:40:40 -06:00
|
|
|
|
</div>
|
|
|
|
|
</ActionTooltip>
|
|
|
|
|
</LongPressable>
|
|
|
|
|
<TipPopover target={ref.current} show={tipShow} handleClose={() => setTipShow(false)} />
|
|
|
|
|
<UpvotePopover target={ref.current} show={voteShow} handleClose={() => setVoteShow(false)} />
|
|
|
|
|
</div>}
|
2021-04-22 17:14:32 -05:00
|
|
|
|
</LightningConsumer>
|
|
|
|
|
)
|
|
|
|
|
}
|