183 lines
6.3 KiB
JavaScript
Raw Normal View History

2021-04-14 18:56:29 -05:00
import itemStyles from './item.module.css'
import styles from './comment.module.css'
import Text from './text'
import Link from 'next/link'
import Reply from './reply'
2021-06-24 18:56:01 -05:00
import { useEffect, useRef, useState } from 'react'
2021-04-18 13:50:04 -05:00
import { timeSince } from '../lib/time'
2021-04-22 17:14:32 -05:00
import UpVote from './upvote'
2021-04-30 16:42:51 -05:00
import Eye from '../svgs/eye-fill.svg'
import EyeClose from '../svgs/eye-close-line.svg'
2021-06-24 18:56:01 -05:00
import { useRouter } from 'next/router'
2021-08-10 17:59:06 -05:00
import CommentEdit from './comment-edit'
2021-08-11 15:34:10 -05:00
import Countdown from './countdown'
import { NOFOLLOW_LIMIT } from '../lib/constants'
2021-04-14 18:56:29 -05:00
function Parent ({ item, rootText }) {
2021-04-15 14:41:02 -05:00
const ParentFrag = () => (
<>
<span> \ </span>
<Link href={`/items/${item.parentId}`} passHref>
2021-06-24 18:56:01 -05:00
<a onClick={e => e.stopPropagation()} className='text-reset'>parent</a>
2021-04-15 14:41:02 -05:00
</Link>
</>
)
2021-07-07 19:15:27 -05:00
if (!item.root) {
2021-04-15 14:41:02 -05:00
return <ParentFrag />
}
return (
<>
2021-07-07 19:15:27 -05:00
{Number(item.root.id) !== Number(item.parentId) && <ParentFrag />}
2021-04-15 14:41:02 -05:00
<span> \ </span>
2021-07-07 19:15:27 -05:00
<Link href={`/items/${item.root.id}`} passHref>
<a onClick={e => e.stopPropagation()} className='text-reset'>{rootText || 'on:'} {item.root.title}</a>
2021-04-15 14:41:02 -05:00
</Link>
</>
)
}
2021-09-23 12:42:00 -05:00
export default function Comment ({
item, children, replyOpen, includeParent,
2021-09-24 16:28:21 -05:00
rootText, noComments, noReply
2021-09-23 12:42:00 -05:00
}) {
2021-08-10 17:59:06 -05:00
const [edit, setEdit] = useState()
2021-04-30 16:42:51 -05:00
const [collapse, setCollapse] = useState(false)
2021-06-24 18:56:01 -05:00
const ref = useRef(null)
const router = useRouter()
2021-11-27 12:01:02 -06:00
const mine = item.mine
2021-08-10 17:59:06 -05:00
const editThreshold = new Date(item.createdAt).getTime() + 10 * 60000
const [canEdit, setCanEdit] =
useState(mine && (Date.now() < editThreshold))
2021-06-24 18:56:01 -05:00
useEffect(() => {
if (Number(router.query.commentId) === Number(item.id)) {
ref.current.scrollIntoView()
2021-08-17 18:59:22 -05:00
ref.current.classList.add('flash-it')
2021-06-24 18:56:01 -05:00
}
2021-10-27 13:35:26 -05:00
setCollapse(localStorage.getItem(`commentCollapse:${item.id}`))
2021-06-24 18:56:01 -05:00
}, [item])
2021-04-14 18:56:29 -05:00
2021-10-27 13:26:34 -05:00
const op = item.root.user.name === item.user.name
2021-04-14 18:56:29 -05:00
return (
2021-11-09 16:43:56 -06:00
<div
2021-08-17 18:59:22 -05:00
ref={ref} className={includeParent ? '' : `${styles.comment} ${collapse ? styles.collapsed : ''}`}
2021-06-24 18:56:01 -05:00
>
2021-04-28 14:30:14 -05:00
<div className={`${itemStyles.item} ${styles.item}`}>
2021-09-10 16:13:52 -05:00
<UpVote item={item} className={styles.upvote} />
2021-04-30 16:42:51 -05:00
<div className={`${itemStyles.hunk} ${styles.hunk}`}>
<div className='d-flex align-items-center'>
<div className={`${itemStyles.other} ${styles.other}`}>
2021-11-26 13:24:51 -06:00
<span title={`${item.sats} upvotes \\ ${item.tips} tipped`}>{item.sats + item.tips} sats</span>
2021-04-30 16:42:51 -05:00
<span> \ </span>
2021-09-10 16:13:52 -05:00
{item.boost > 0 &&
<>
<span>{item.boost} boost</span>
</>}
2021-04-30 16:42:51 -05:00
<Link href={`/items/${item.id}`} passHref>
2021-06-24 18:56:01 -05:00
<a onClick={e => e.stopPropagation()} className='text-reset'>{item.ncomments} replies</a>
2021-04-30 16:42:51 -05:00
</Link>
<span> \ </span>
<Link href={`/${item.user.name}`} passHref>
2021-10-29 11:02:26 -05:00
<a onClick={e => e.stopPropagation()}>@{item.user.name}<span className='text-boost font-weight-bold'>{op && ' OP'}</span></a>
2021-04-30 16:42:51 -05:00
</Link>
<span> </span>
<span>{timeSince(new Date(item.createdAt))}</span>
{includeParent && <Parent item={item} rootText={rootText} />}
2021-09-23 15:09:07 -05:00
{canEdit &&
<>
<span> \ </span>
<div
className={styles.edit}
onClick={() => setEdit(!edit)}
>
{edit ? 'cancel' : 'edit'}
<Countdown
date={editThreshold}
onComplete={() => {
setCanEdit(false)
}}
/>
</div>
</>}
2021-04-30 16:42:51 -05:00
</div>
{!includeParent && (collapse
2021-10-27 13:35:26 -05:00
? <Eye
className={styles.collapser} height={10} width={10} onClick={() => {
setCollapse(false)
localStorage.removeItem(`commentCollapse:${item.id}`)
}}
/>
: <EyeClose
className={styles.collapser} height={10} width={10} onClick={() => {
setCollapse(true)
localStorage.setItem(`commentCollapse:${item.id}`, 'yep')
}}
/>)}
2021-04-14 18:56:29 -05:00
</div>
2021-08-10 17:59:06 -05:00
{edit
? (
2021-09-23 15:09:07 -05:00
<CommentEdit
comment={item}
onSuccess={() => {
setEdit(!edit)
setCanEdit(mine && (Date.now() < editThreshold))
}}
/>
2021-08-10 17:59:06 -05:00
)
: (
<div className={styles.text}>
<Text nofollow={item.sats + item.boost < NOFOLLOW_LIMIT}>{item.text}</Text>
2021-08-10 17:59:06 -05:00
</div>
)}
2021-04-14 18:56:29 -05:00
</div>
</div>
2021-08-18 13:59:30 -05:00
<div className={`${styles.children}`}>
2021-09-24 16:28:21 -05:00
{!noReply &&
<Reply
parentId={item.id} replyOpen={replyOpen}
/>}
2021-04-28 14:30:14 -05:00
{children}
<div className={`${styles.comments} ml-sm-1 ml-md-3`}>
{item.comments && !noComments
? item.comments.map((item) => (
<Comment key={item.id} item={item} />
))
: null}
</div>
</div>
2021-11-09 16:43:56 -06:00
</div>
2021-04-22 17:14:32 -05:00
)
}
export function CommentSkeleton ({ skeletonChildren }) {
return (
2021-04-28 17:52:03 -05:00
<div className={styles.comment}>
2021-04-22 17:14:32 -05:00
<div className={`${itemStyles.item} ${itemStyles.skeleton} ${styles.item} ${styles.skeleton}`}>
<UpVote className={styles.upvote} />
<div className={`${itemStyles.hunk} ${styles.hunk}`}>
<div className={itemStyles.other}>
2021-04-28 17:52:03 -05:00
<span className={`${itemStyles.otherItem} clouds`} />
<span className={`${itemStyles.otherItem} clouds`} />
2021-04-22 17:14:32 -05:00
<span className={`${itemStyles.otherItem} clouds`} />
<span className={`${itemStyles.otherItem} ${itemStyles.otherItemLonger} clouds`} />
</div>
<div className={`${styles.text} clouds`} />
2021-04-14 18:56:29 -05:00
</div>
2021-04-22 17:14:32 -05:00
</div>
2021-04-28 17:52:03 -05:00
<div className={`${itemStyles.children} ${styles.children} ${styles.skeleton}`}>
<div className={styles.replyPadder}>
<div className={`${itemStyles.other} ${styles.reply} clouds`} />
</div>
<div className={`${styles.comments} ml-sm-1 ml-md-3`}>
2021-05-05 13:13:14 -05:00
{skeletonChildren
? <CommentSkeleton skeletonChildren={skeletonChildren - 1} />
2021-04-17 13:15:18 -05:00
: null}
</div>
2021-04-14 18:56:29 -05:00
</div>
2021-04-28 17:52:03 -05:00
</div>
2021-04-14 18:56:29 -05:00
)
}