import Link from 'next/link' import styles from './item.module.css' import UpVote from './upvote' import { useRef } from 'react' import { USER_ID, UNKNOWN_LINK_REL } from '@/lib/constants' import Pin from '@/svgs/pushpin-fill.svg' import reactStringReplace from 'react-string-replace' import PollIcon from '@/svgs/bar-chart-horizontal-fill.svg' import BountyIcon from '@/svgs/bounty-bag.svg' import ActionTooltip from './action-tooltip' import ImageIcon from '@/svgs/image-fill.svg' import VideoIcon from '@/svgs/video-on-fill.svg' import { numWithUnits } from '@/lib/format' import ItemInfo from './item-info' import Prism from '@/svgs/prism.svg' import { commentsViewedAt } from '@/lib/new-comments' import { useRouter } from 'next/router' import { Badge } from 'react-bootstrap' import AdIcon from '@/svgs/advertisement-fill.svg' import { DownZap } from './dont-link-this' import { timeLeft } from '@/lib/time' import classNames from 'classnames' import removeMd from 'remove-markdown' import { decodeProxyUrl, IMGPROXY_URL_REGEXP, parseInternalLinks } from '@/lib/url' import ItemPopover from './item-popover' import { useMe } from './me' import Boost from './boost-button' function onItemClick (e, router, item) { const viewedAt = commentsViewedAt(item) if (viewedAt) { e.preventDefault() if (e.ctrlKey || e.metaKey) { window.open( `/items/${item.id}`, '_blank', 'noopener,noreferrer' ) } else { router.push( `/items/${item.id}?commentsViewedAt=${viewedAt}`, `/items/${item.id}`) } } } export function SearchTitle ({ title }) { return reactStringReplace(title, /\*\*\*([^*]+)\*\*\*/g, (match, i) => { return {match} }) } function mediaType ({ url, imgproxyUrls }) { const { me } = useMe() const src = IMGPROXY_URL_REGEXP.test(url) ? decodeProxyUrl(url) : url if (!imgproxyUrls?.[src] || me?.privates?.showImagesAndVideos === false || // we don't proxy videos even if we have thumbnails (me?.privates?.imgproxyOnly && imgproxyUrls?.[src]?.video)) return return imgproxyUrls?.[src]?.video ? 'video' : 'image' } function ItemLink ({ url, rel }) { try { const { linkText } = parseInternalLinks(url) if (linkText) { return ( {linkText} ) } return ( // eslint-disable-next-line {url.replace(/(^https?:|^)\/\//, '')} ) } catch { return null } } export default function Item ({ item, rank, belowTitle, right, full, children, itemClassName, onQuoteReply, pinnable }) { const titleRef = useRef() const router = useRouter() const media = mediaType({ url: item.url, imgproxyUrls: item.imgproxyUrls }) const MediaIcon = media === 'video' ? VideoIcon : ImageIcon return ( <> {rank ? (
{rank}
) :
}
{item.position && (pinnable || !item.subName) ? : item.mine ? : item.meDontLikeSats > item.meSats ? : Number(item.user?.id) === USER_ID.ad ? : }
onItemClick(e, router, item)} ref={titleRef} className={`${styles.title} text-reset me-2`} > {item.searchTitle ? : item.title} {item.pollCost && } {item.bounty > 0 && } {item.forwards?.length > 0 && } {media && } {item.url && !media && }
AD} /> {belowTitle}
{right}
{children && (
{children}
)} ) } export function ItemSummary ({ item }) { const router = useRouter() const link = ( onItemClick(e, router, item)} className={`${item.title && styles.title} ${styles.summaryText} text-reset me-2`} > {item.title ?? removeMd(item.text)} ) const info = ( AD} /> ) return (
{item.title ? ( <> {link} {info} ) : ( <> {info} {link} )}
) } export function ItemSkeleton ({ rank, children, showUpvote = true }) { return ( <> {rank ? (
{rank}
) :
}
{showUpvote && }
{children && (
{children}
)} ) } function PollIndicator ({ item }) { const hasExpiration = !!item.pollExpiresAt const timeRemaining = timeLeft(new Date(item.pollExpiresAt)) const isActive = !hasExpiration || !!timeRemaining return ( ) }