stacker.news/components/upvote.js

219 lines
6.6 KiB
JavaScript
Raw Normal View History

import UpBolt from '@/svgs/bolt.svg'
2021-04-22 22:14:32 +00:00
import styles from './upvote.module.css'
import { gql, useMutation } from '@apollo/client'
2021-07-08 18:42:57 +00:00
import ActionTooltip from './action-tooltip'
2023-12-27 02:27:52 +00:00
import ItemAct, { useAct, useZap } from './item-act'
2021-09-12 16:55:38 +00:00
import { useMe } from './me'
import getColor from '@/lib/rainbow'
2023-10-06 20:01:51 +00:00
import { useCallback, useMemo, useRef, useState } from 'react'
import LongPressable from './long-pressable'
2023-07-24 18:35:05 +00:00
import Overlay from 'react-bootstrap/Overlay'
import Popover from 'react-bootstrap/Popover'
import { useShowModal } from './modal'
import { numWithUnits } from '@/lib/format'
import { Dropdown } from 'react-bootstrap'
2021-12-09 20:40:40 +00:00
2022-04-12 21:09:12 +00:00
const UpvotePopover = ({ target, show, handleClose }) => {
const me = useMe()
return (
<Overlay
show={show}
target={target}
placement='right'
>
<Popover id='popover-basic'>
<Popover.Header className='d-flex justify-content-between alert-dismissible' as='h4'>Zapping
<button type='button' className='btn-close' onClick={handleClose}><span className='visually-hidden-focusable'>Close alert</span></button>
</Popover.Header>
2023-07-24 18:35:05 +00:00
<Popover.Body>
<div className='mb-2'>Press the bolt again to zap {me?.privates?.tipDefault || 1} more sat{me?.privates?.tipDefault > 1 ? 's' : ''}.</div>
2023-06-19 18:21:55 +00:00
<div>Repeatedly press the bolt to zap more sats.</div>
2023-07-24 18:35:05 +00:00
</Popover.Body>
2022-04-12 21:09:12 +00:00
</Popover>
</Overlay>
)
}
2021-12-09 20:40:40 +00:00
const TipPopover = ({ target, show, handleClose }) => (
<Overlay
show={show}
target={target}
placement='right'
>
<Popover id='popover-basic'>
<Popover.Header className='d-flex justify-content-between alert-dismissible' as='h4'>Press and hold
<button type='button' className='btn-close' onClick={handleClose}><span className='visually-hidden-focusable'>Close alert</span></button>
</Popover.Header>
2023-07-24 18:35:05 +00:00
<Popover.Body>
2023-06-19 18:21:55 +00:00
<div className='mb-2'>Press and hold bolt to zap a custom amount.</div>
<div>As you zap more, the bolt color follows the rainbow.</div>
2023-07-24 18:35:05 +00:00
</Popover.Body>
2021-12-09 20:40:40 +00:00
</Popover>
</Overlay>
)
export function DropdownItemUpVote ({ item }) {
const showModal = useShowModal()
return (
<Dropdown.Item
onClick={async () => {
showModal(onClose =>
2023-12-27 02:27:52 +00:00
<ItemAct onClose={onClose} itemId={item.id} />)
}}
>
<span className='text-success'>zap</span>
</Dropdown.Item>
)
}
export const nextTip = (meSats, { tipDefault, turboTipping }) => {
// what should our next tip be?
if (!turboTipping) return (tipDefault || 1)
let sats = tipDefault || 1
if (turboTipping) {
while (meSats >= sats) {
sats *= 10
}
// deduct current sats since turbo tipping is about total zap not making the next zap 10x
sats -= meSats
}
return sats
}
2023-12-27 02:27:52 +00:00
export default function UpVote ({ item, className }) {
const showModal = useShowModal()
const [voteShow, _setVoteShow] = useState(false)
const [tipShow, _setTipShow] = useState(false)
const ref = useRef()
const me = useMe()
const [hover, setHover] = useState(false)
const [setWalkthrough] = useMutation(
gql`
mutation setWalkthrough($upvotePopover: Boolean, $tipPopover: Boolean) {
setWalkthrough(upvotePopover: $upvotePopover, tipPopover: $tipPopover)
}`
)
const setVoteShow = useCallback((yes) => {
if (!me) return
// if they haven't seen the walkthrough and they have sats
if (yes && !me.privates?.upvotePopover && me.privates?.sats) {
_setVoteShow(true)
}
if (voteShow && !yes) {
_setVoteShow(false)
setWalkthrough({ variables: { upvotePopover: true } })
}
}, [me, voteShow, setWalkthrough])
const setTipShow = useCallback((yes) => {
if (!me) return
// if we want to show it, yet we still haven't shown
if (yes && !me.privates?.tipPopover && me.privates?.sats) {
_setTipShow(true)
}
// if it's currently showing and we want to hide it
if (tipShow && !yes) {
_setTipShow(false)
setWalkthrough({ variables: { tipPopover: true } })
}
}, [me, tipShow, setWalkthrough])
2023-12-26 22:51:47 +00:00
const [act] = useAct()
2023-12-27 02:27:52 +00:00
const zap = useZap()
2023-08-28 14:40:29 +00:00
const disabled = useMemo(() => item?.mine || item?.meForward || item?.deletedAt,
[item?.mine, item?.meForward, item?.deletedAt])
const [meSats, overlayText, color, nextColor] = useMemo(() => {
2023-12-27 02:27:52 +00:00
const meSats = (item?.meSats || item?.meAnonSats || 0)
2022-12-09 19:25:38 +00:00
// what should our next tip be?
const sats = nextTip(meSats, { ...me?.privates })
2021-09-12 16:55:38 +00:00
return [
meSats, me ? numWithUnits(sats, { abbreviate: false }) : 'zap it',
getColor(meSats), getColor(meSats + sats)]
2023-12-27 02:27:52 +00:00
}, [item?.meSats, item?.meAnonSats, me?.privates?.tipDefault, me?.privates?.turboDefault])
2023-01-12 23:53:09 +00:00
const handleModalClosed = () => {
setHover(false)
}
const handleLongPress = (e) => {
if (!item) return
// we can't tip ourselves
if (disabled) {
return
}
setTipShow(false)
showModal(onClose =>
<ItemAct onClose={onClose} itemId={item.id} />, { onClose: handleModalClosed })
}
const handleShortPress = () => {
if (me) {
if (!item) return
// we can't tip ourselves
if (disabled) {
return
}
if (meSats) {
setVoteShow(false)
} else {
setTipShow(true)
}
zap({ item, me })
} else {
showModal(onClose => <ItemAct onClose={onClose} itemId={item.id} act={act} />, { onClose: handleModalClosed })
}
}
2021-04-22 22:14:32 +00:00
return (
2023-12-27 02:27:52 +00:00
<div ref={ref} className='upvoteParent'>
<LongPressable
onLongPress={handleLongPress}
onShortPress={handleShortPress}
2023-12-27 02:27:52 +00:00
>
<ActionTooltip notForm disable={disabled} overlayText={overlayText}>
<div
className={`${disabled ? styles.noSelfTips : ''} ${styles.upvoteWrapper}`}
2021-12-09 20:40:40 +00:00
>
2023-12-27 02:27:52 +00:00
<UpBolt
2024-03-02 00:32:40 +00:00
onPointerEnter={() => setHover(true)}
onMouseLeave={() => setHover(false)}
2024-03-02 00:32:40 +00:00
onTouchEnd={() => setHover(false)}
2023-12-27 02:27:52 +00:00
width={26}
height={26}
className={
2021-12-09 20:40:40 +00:00
`${styles.upvote}
${className || ''}
2023-01-12 23:53:09 +00:00
${disabled ? styles.noSelfTips : ''}
${meSats ? styles.voted : ''}`
2021-12-09 20:40:40 +00:00
}
style={meSats || hover
2023-12-27 02:27:52 +00:00
? {
fill: hover ? nextColor : color,
2024-03-13 16:48:16 +00:00
filter: `drop-shadow(0 0 6px ${hover ? nextColor : color}90)`
2023-12-27 02:27:52 +00:00
}
: undefined}
/>
</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 22:14:32 +00:00
)
}