refactor embeds to be reused (#1368)

* refactor embeds to be reused

* adjust the meaning of settings for embeds

* add wavlake embed (close #1359)

* add spotify embed (closes #1360)

* fix 'format' appearing in srcSet

* add nostr embed

* refine nostr embed

* Update components/media-or-link.js

Co-authored-by: ekzyis <ek@stacker.news>

* Update pages/settings/index.js

Co-authored-by: ekzyis <ek@stacker.news>

* ek suggestions

---------

Co-authored-by: ekzyis <ek@stacker.news>
This commit is contained in:
Keyan 2024-09-07 12:07:10 -05:00 committed by GitHub
parent f05b29717a
commit 15b038cd78
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 455 additions and 286 deletions

View File

@ -9,10 +9,7 @@ import styles from '@/styles/item.module.css'
import itemStyles from './item.module.css' import itemStyles from './item.module.css'
import { useMe } from './me' import { useMe } from './me'
import Button from 'react-bootstrap/Button' import Button from 'react-bootstrap/Button'
import { TwitterTweetEmbed } from 'react-twitter-embed' import { useEffect } from 'react'
import YouTube from 'react-youtube'
import useDarkMode from './dark-mode'
import { useEffect, useState } from 'react'
import Poll from './poll' import Poll from './poll'
import { commentsViewed } from '@/lib/new-comments' import { commentsViewed } from '@/lib/new-comments'
import Related from './related' import Related from './related'
@ -22,7 +19,7 @@ import Share from './share'
import Toc from './table-of-contents' import Toc from './table-of-contents'
import Link from 'next/link' import Link from 'next/link'
import { RootProvider } from './root' import { RootProvider } from './root'
import { IMGPROXY_URL_REGEXP, parseEmbedUrl } from '@/lib/url' import { decodeProxyUrl, IMGPROXY_URL_REGEXP } from '@/lib/url'
import { numWithUnits } from '@/lib/format' import { numWithUnits } from '@/lib/format'
import { useQuoteReply } from './use-quote-reply' import { useQuoteReply } from './use-quote-reply'
import { UNKNOWN_LINK_REL } from '@/lib/constants' import { UNKNOWN_LINK_REL } from '@/lib/constants'
@ -50,91 +47,11 @@ function BioItem ({ item, handleClick }) {
) )
} }
function TweetSkeleton () { function ItemEmbed ({ url, imgproxyUrls }) {
return ( const src = IMGPROXY_URL_REGEXP.test(url) ? decodeProxyUrl(url) : url
<div className={styles.tweetsSkeleton}> const srcSet = imgproxyUrls?.[url]
<div className={styles.tweetSkeleton}>
<div className={`${styles.img} clouds`} />
<div className={styles.content1}>
<div className={`${styles.line} clouds`} />
<div className={`${styles.line} clouds`} />
<div className={`${styles.line} clouds`} />
</div>
</div>
</div>
)
}
function ItemEmbed ({ item }) { return <MediaOrLink src={src} srcSet={srcSet} topLevel linkFallback={false} />
const [darkMode] = useDarkMode()
const [overflowing, setOverflowing] = useState(false)
const [show, setShow] = useState(false)
// This Twitter embed could use similar logic to the video embeds below
const twitter = item.url?.match(/^https?:\/\/(?:twitter|x)\.com\/(?:#!\/)?\w+\/status(?:es)?\/(?<id>\d+)/)
if (twitter?.groups?.id) {
return (
<div className={`${styles.twitterContainer} ${show ? '' : styles.twitterContained}`}>
<TwitterTweetEmbed tweetId={twitter.groups.id} options={{ theme: darkMode ? 'dark' : 'light', width: '550px' }} key={darkMode ? '1' : '2'} placeholder={<TweetSkeleton />} onLoad={() => setOverflowing(true)} />
{overflowing && !show &&
<Button size='lg' variant='info' className={styles.twitterShowFull} onClick={() => setShow(true)}>
show full tweet
</Button>}
</div>
)
}
const { provider, id, meta } = parseEmbedUrl(item.url)
if (provider === 'youtube') {
return (
<div className={styles.videoWrapper}>
<YouTube
videoId={id} className={styles.videoContainer} opts={{
playerVars: {
start: meta?.start || 0
}
}}
/>
</div>
)
}
if (provider === 'rumble') {
return (
<div className={styles.videoWrapper}>
<div className={styles.videoContainer}>
<iframe
title='Rumble Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
if (provider === 'peertube') {
return (
<div className={styles.videoWrapper}>
<div className={styles.videoContainer}>
<iframe
title='PeerTube Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
if (item.url?.match(IMGPROXY_URL_REGEXP)) {
return <MediaOrLink src={item.url} rel={item.rel ?? UNKNOWN_LINK_REL} />
}
return null
} }
function FwdUsers ({ forwards }) { function FwdUsers ({ forwards }) {
@ -174,7 +91,7 @@ function TopLevelItem ({ item, noReply, ...props }) {
> >
<article className={styles.fullItemContainer} ref={textRef}> <article className={styles.fullItemContainer} ref={textRef}>
{item.text && <ItemText item={item} />} {item.text && <ItemText item={item} />}
{item.url && <ItemEmbed item={item} />} {item.url && !item.outlawed && <ItemEmbed url={item.url} imgproxyUrls={item.imgproxyUrls} />}
{item.poll && <Poll item={item} />} {item.poll && <Poll item={item} />}
{item.bounty && {item.bounty &&
<div className='fw-bold mt-2'> <div className='fw-bold mt-2'>

View File

@ -9,6 +9,7 @@ import PollIcon from '@/svgs/bar-chart-horizontal-fill.svg'
import BountyIcon from '@/svgs/bounty-bag.svg' import BountyIcon from '@/svgs/bounty-bag.svg'
import ActionTooltip from './action-tooltip' import ActionTooltip from './action-tooltip'
import ImageIcon from '@/svgs/image-fill.svg' import ImageIcon from '@/svgs/image-fill.svg'
import VideoIcon from '@/svgs/video-on-fill.svg'
import { numWithUnits } from '@/lib/format' import { numWithUnits } from '@/lib/format'
import ItemInfo from './item-info' import ItemInfo from './item-info'
import Prism from '@/svgs/prism.svg' import Prism from '@/svgs/prism.svg'
@ -20,6 +21,9 @@ import { DownZap } from './dont-link-this'
import { timeLeft } from '@/lib/time' import { timeLeft } from '@/lib/time'
import classNames from 'classnames' import classNames from 'classnames'
import removeMd from 'remove-markdown' import removeMd from 'remove-markdown'
import { decodeProxyUrl, IMGPROXY_URL_REGEXP, parseInternalLinks } from '@/lib/url'
import ItemPopover from './item-popover'
import { useMe } from './me'
function onItemClick (e, router, item) { function onItemClick (e, router, item) {
const viewedAt = commentsViewedAt(item) const viewedAt = commentsViewedAt(item)
@ -45,6 +49,41 @@ export function SearchTitle ({ title }) {
}) })
} }
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 (
<ItemPopover id={linkText.replace('#', '').split('/')[0]}>
<Link href={url} className={styles.link}>{linkText}</Link>
</ItemPopover>
)
}
return (
// eslint-disable-next-line
<a
className={styles.link} target='_blank' href={url}
rel={rel ?? UNKNOWN_LINK_REL}
>
{url.replace(/(^https?:|^)\/\//, '')}
</a>
)
} catch {
return null
}
}
export default function Item ({ export default function Item ({
item, rank, belowTitle, right, full, children, itemClassName, item, rank, belowTitle, right, full, children, itemClassName,
onQuoteReply, pinnable onQuoteReply, pinnable
@ -52,7 +91,8 @@ export default function Item ({
const titleRef = useRef() const titleRef = useRef()
const router = useRouter() const router = useRouter()
const image = item.url && item.url.startsWith(process.env.NEXT_PUBLIC_IMGPROXY_URL) const media = mediaType({ url: item.url, imgproxyUrls: item.imgproxyUrls })
const MediaIcon = media === 'video' ? VideoIcon : ImageIcon
return ( return (
<> <>
@ -87,16 +127,9 @@ export default function Item ({
</ActionTooltip> </ActionTooltip>
</span>} </span>}
{item.forwards?.length > 0 && <span className={styles.icon}><Prism className='fill-grey ms-1' height={14} width={14} /></span>} {item.forwards?.length > 0 && <span className={styles.icon}><Prism className='fill-grey ms-1' height={14} width={14} /></span>}
{image && <span className={styles.icon}><ImageIcon className='fill-grey ms-2' height={16} width={16} /></span>} {media && <span className={styles.icon}><MediaIcon className='fill-grey ms-2' height={16} width={16} /></span>}
</Link> </Link>
{item.url && !image && {item.url && !media && <ItemLink url={item.url} rel={UNKNOWN_LINK_REL} />}
// eslint-disable-next-line
<a
className={styles.link} target='_blank' href={item.url}
rel={item.rel ?? UNKNOWN_LINK_REL}
>
{item.url.replace(/(^https?:|^)\/\//, '')}
</a>}
</div> </div>
<ItemInfo <ItemInfo
full={full} item={item} full={full} item={item}

View File

@ -1,11 +1,14 @@
import styles from './text.module.css' import styles from './text.module.css'
import { useState, useEffect, useMemo, useCallback, memo } from 'react' import { useState, useEffect, useMemo, useCallback, memo, useRef } from 'react'
import { decodeProxyUrl, IMGPROXY_URL_REGEXP, MEDIA_DOMAIN_REGEXP } from '@/lib/url' import { decodeProxyUrl, IMGPROXY_URL_REGEXP, MEDIA_DOMAIN_REGEXP, parseEmbedUrl } from '@/lib/url'
import { useShowModal } from './modal' import { useShowModal } from './modal'
import { useMe } from './me' import { useMe } from './me'
import { Dropdown } from 'react-bootstrap' import { Button, Dropdown } from 'react-bootstrap'
import { UNKNOWN_LINK_REL } from '@/lib/constants' import { UNKNOWN_LINK_REL } from '@/lib/constants'
import classNames from 'classnames' import classNames from 'classnames'
import { TwitterTweetEmbed } from 'react-twitter-embed'
import YouTube from 'react-youtube'
import useDarkMode from './dark-mode'
function LinkRaw ({ href, children, src, rel }) { function LinkRaw ({ href, children, src, rel }) {
const isRawURL = /^https?:\/\//.test(children?.[0]) const isRawURL = /^https?:\/\//.test(children?.[0])
@ -46,7 +49,7 @@ const Media = memo(function Media ({ src, bestResSrc, srcSet, sizes, width, heig
) )
}) })
export default function MediaOrLink (props) { export default function MediaOrLink ({ linkFallback = true, ...props }) {
const media = useMediaHelper(props) const media = useMediaHelper(props)
const [error, setError] = useState(false) const [error, setError] = useState(false)
const showModal = useShowModal() const showModal = useShowModal()
@ -78,28 +81,45 @@ export default function MediaOrLink (props) {
if (!media.src) return null if (!media.src) return null
if (!error && (media.image || media.video)) { if (!error) {
return ( if (media.image || media.video) {
<Media return (
{...media} onClick={handleClick} onError={handleError} <Media
/> {...media} onClick={handleClick} onError={handleError}
) />
)
}
if (media.embed) {
return (
<Embed
{...media.embed} src={media.src}
className={media.className} onError={handleError} topLevel={props.topLevel}
/>
)
}
} }
return <LinkRaw {...props} /> if (linkFallback) {
return <LinkRaw {...props} />
}
return null
} }
// determines how the media should be displayed given the params, me settings, and editor tab // determines how the media should be displayed given the params, me settings, and editor tab
const useMediaHelper = ({ src, srcSet: { dimensions, video, ...srcSetObj } = {}, topLevel, tab }) => { export const useMediaHelper = ({ src, srcSet: srcSetIntital, topLevel, tab }) => {
const me = useMe() const me = useMe()
const trusted = useMemo(() => !!srcSetObj || IMGPROXY_URL_REGEXP.test(src) || MEDIA_DOMAIN_REGEXP.test(src), [!!srcSetObj, src]) const trusted = useMemo(() => !!srcSetIntital || IMGPROXY_URL_REGEXP.test(src) || MEDIA_DOMAIN_REGEXP.test(src), [!!srcSetIntital, src])
const { dimensions, video, format, ...srcSetObj } = srcSetIntital || {}
const [isImage, setIsImage] = useState(!video && trusted) const [isImage, setIsImage] = useState(!video && trusted)
const [isVideo, setIsVideo] = useState(video) const [isVideo, setIsVideo] = useState(video)
const showMedia = useMemo(() => tab === 'preview' || me?.privates?.showImagesAndVideos !== false, [tab, me?.privates?.showImagesAndVideos]) const showMedia = useMemo(() => tab === 'preview' || me?.privates?.showImagesAndVideos !== false, [tab, me?.privates?.showImagesAndVideos])
const embed = useMemo(() => parseEmbedUrl(src), [src])
useEffect(() => { useEffect(() => {
// don't load the video at all if use doesn't want these // don't load the video at all if user doesn't want these
if (!showMedia || isVideo || isImage) return if (!showMedia || isVideo || isImage || embed) return
// make sure it's not a false negative by trying to load URL as <img> // make sure it's not a false negative by trying to load URL as <img>
const img = new window.Image() const img = new window.Image()
img.onload = () => setIsImage(true) img.onload = () => setIsImage(true)
@ -114,7 +134,7 @@ const useMediaHelper = ({ src, srcSet: { dimensions, video, ...srcSetObj } = {},
video.onloadeddata = null video.onloadeddata = null
video.src = '' video.src = ''
} }
}, [src, setIsImage, setIsVideo, showMedia, isVideo]) }, [src, setIsImage, setIsVideo, showMedia, isVideo, embed])
const srcSet = useMemo(() => { const srcSet = useMemo(() => {
if (Object.keys(srcSetObj).length === 0) return undefined if (Object.keys(srcSetObj).length === 0) return undefined
@ -164,7 +184,177 @@ const useMediaHelper = ({ src, srcSet: { dimensions, video, ...srcSetObj } = {},
width, width,
height, height,
className: classNames(topLevel && styles.topLevel), className: classNames(topLevel && styles.topLevel),
image: (!me?.privates?.imgproxyOnly || trusted) && showMedia && isImage && !isVideo, image: (!me?.privates?.imgproxyOnly || trusted) && showMedia && isImage && !isVideo && !embed,
video: !me?.privates?.imgproxyOnly && showMedia && isVideo video: !me?.privates?.imgproxyOnly && showMedia && isVideo && !embed,
embed: !me?.privates?.imgproxyOnly && showMedia && embed
} }
} }
function TweetSkeleton ({ className }) {
return (
<div className={classNames(styles.tweetsSkeleton, className)}>
<div className={styles.tweetSkeleton}>
<div className={`${styles.img} clouds`} />
<div className={styles.content1}>
<div className={`${styles.line} clouds`} />
<div className={`${styles.line} clouds`} />
<div className={`${styles.line} clouds`} />
</div>
</div>
</div>
)
}
export const useFrameHeight = (
iframeRef
) => {
const [height, setHeight] = useState(0)
const iframeCurrent = iframeRef.current
useEffect(() => {
const setHeightFromIframe = (e) => {
if (e.origin !== 'https://njump.me' || !e?.data?.height) return
setHeight(e.data.height)
}
window?.addEventListener('message', setHeightFromIframe)
return () => {
window?.removeEventListener('message', setHeightFromIframe)
}
}, [iframeCurrent])
return height
}
export const NostrEmbed = memo(function NostrEmbed ({ src, className, topLevel, id }) {
const [show, setShow] = useState(false)
const iframeRef = useRef(null)
const frameHeight = useFrameHeight(iframeRef)
return (
<div className={classNames(styles.nostrContainer, !show && styles.twitterContained, className)}>
<iframe
ref={iframeRef}
src={`https://njump.me/${id}?embed=yes`}
width={topLevel ? '550px' : '350px'}
style={{ maxWidth: '100%' }}
height={frameHeight ? `${frameHeight}px` : topLevel ? '200px' : '150px'}
frameBorder='0'
sandbox='allow-scripts allow-same-origin'
allow=''
/>
{!show &&
<Button size='md' variant='info' className={styles.twitterShowFull} onClick={() => setShow(true)}>
<div>show full note</div>
<small className='fw-normal fst-italic'>or other stuff</small>
</Button>}
</div>
)
})
export const Embed = memo(function Embed ({ src, provider, id, meta, className, topLevel, onError }) {
const [darkMode] = useDarkMode()
const [overflowing, setOverflowing] = useState(true)
const [show, setShow] = useState(false)
// This Twitter embed could use similar logic to the video embeds below
if (provider === 'twitter') {
return (
<div className={classNames(styles.twitterContainer, !show && styles.twitterContained, className)}>
<TwitterTweetEmbed
tweetId={id}
options={{ theme: darkMode ? 'dark' : 'light', width: topLevel ? '550px' : '350px' }}
key={darkMode ? '1' : '2'}
placeholder={<TweetSkeleton className={className} />}
onLoad={() => setOverflowing(true)}
/>
{overflowing && !show &&
<Button size='lg' variant='info' className={styles.twitterShowFull} onClick={() => setShow(true)}>
show full tweet
</Button>}
</div>
)
}
if (provider === 'nostr') {
return (
<NostrEmbed src={src} className={className} topLevel={topLevel} id={id} />
)
}
if (provider === 'wavlake') {
return (
<div className={classNames(styles.wavlakeWrapper, className)}>
<iframe
src={`https://embed.wavlake.com/track/${id}`} width='100%' height='380' frameBorder='0'
allow='encrypted-media'
sandbox='allow-scripts'
/>
</div>
)
}
if (provider === 'spotify') {
// https://open.spotify.com/track/1KFxcj3MZrpBGiGA8ZWriv?si=f024c3aa52294aa1
// Remove any additional path segments
const url = new URL(src)
url.pathname = url.pathname.replace(/\/intl-\w+\//, '/')
return (
<div className={classNames(styles.spotifyWrapper, className)}>
<iframe
title='Spotify Web Player'
src={`https://open.spotify.com/embed${url.pathname}`}
width='100%'
height='152'
allowfullscreen=''
frameBorder='0'
allow='encrypted-media; clipboard-write;'
style={{ borderRadius: '12px' }}
sandbox='allow-scripts'
/>
</div>
)
}
if (provider === 'youtube') {
return (
<div className={classNames(styles.videoWrapper, className)}>
<YouTube
videoId={id} className={styles.videoContainer} opts={{
playerVars: {
start: meta?.start || 0
}
}}
/>
</div>
)
}
if (provider === 'rumble') {
return (
<div className={classNames(styles.videoWrapper, className)}>
<div className={styles.videoContainer}>
<iframe
title='Rumble Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
if (provider === 'peertube') {
return (
<div className={classNames(styles.videoWrapper, className)}>
<div className={styles.videoContainer}>
<iframe
title='PeerTube Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
return null
})

View File

@ -1,6 +1,5 @@
import styles from './text.module.css' import styles from './text.module.css'
import ReactMarkdown from 'react-markdown' import ReactMarkdown from 'react-markdown'
import YouTube from 'react-youtube'
import gfm from 'remark-gfm' import gfm from 'remark-gfm'
import { LightAsync as SyntaxHighlighter } from 'react-syntax-highlighter' import { LightAsync as SyntaxHighlighter } from 'react-syntax-highlighter'
import atomDark from 'react-syntax-highlighter/dist/cjs/styles/prism/atom-dark' import atomDark from 'react-syntax-highlighter/dist/cjs/styles/prism/atom-dark'
@ -13,7 +12,7 @@ import Thumb from '@/svgs/thumb-up-fill.svg'
import { toString } from 'mdast-util-to-string' import { toString } from 'mdast-util-to-string'
import copy from 'clipboard-copy' import copy from 'clipboard-copy'
import MediaOrLink from './media-or-link' import MediaOrLink from './media-or-link'
import { IMGPROXY_URL_REGEXP, parseInternalLinks, parseEmbedUrl, decodeProxyUrl } from '@/lib/url' import { IMGPROXY_URL_REGEXP, parseInternalLinks, decodeProxyUrl } from '@/lib/url'
import reactStringReplace from 'react-string-replace' import reactStringReplace from 'react-string-replace'
import { rehypeInlineCodeProperty, rehypeStyler } from '@/lib/md' import { rehypeInlineCodeProperty, rehypeStyler } from '@/lib/md'
import { Button } from 'react-bootstrap' import { Button } from 'react-bootstrap'
@ -255,64 +254,11 @@ export default memo(function Text ({ rel, imgproxyUrls, children, tab, itemId, o
// ignore errors like invalid URLs // ignore errors like invalid URLs
} }
const videoWrapperStyles = {
maxWidth: topLevel ? '640px' : '320px',
margin: '0.5rem 0',
paddingRight: '15px'
}
const { provider, id, meta } = parseEmbedUrl(href)
// Youtube video embed
if (provider === 'youtube') {
return (
<div style={videoWrapperStyles}>
<YouTube
videoId={id} className={styles.videoContainer} opts={{
playerVars: {
start: meta?.start || 0
}
}}
/>
</div>
)
}
// Rumble video embed
if (provider === 'rumble') {
return (
<div style={videoWrapperStyles}>
<div className={styles.videoContainer}>
<iframe
title='Rumble Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
if (provider === 'peertube') {
return (
<div style={videoWrapperStyles}>
<div className={styles.videoContainer}>
<iframe
title='PeerTube Video'
allowFullScreen
src={meta?.href}
sandbox='allow-scripts'
/>
</div>
</div>
)
}
// assume the link is an image which will fallback to link if it's not // assume the link is an image which will fallback to link if it's not
return <TextMediaOrLink src={href} rel={rel ?? UNKNOWN_LINK_REL} {...props}>{children}</TextMediaOrLink> return <TextMediaOrLink src={href} rel={rel ?? UNKNOWN_LINK_REL} {...props}>{children}</TextMediaOrLink>
}, },
img: TextMediaOrLink img: TextMediaOrLink
}), [outlawed, rel, topLevel, itemId, Code, P, Heading, Table, TextMediaOrLink]) }), [outlawed, rel, itemId, Code, P, Heading, Table, TextMediaOrLink])
const remarkPlugins = useMemo(() => [gfm, mention, sub], []) const remarkPlugins = useMemo(() => [gfm, mention, sub], [])
const rehypePlugins = useMemo(() => [rehypeInlineCodeProperty, rehypeSuperscript, rehypeSubscript], []) const rehypePlugins = useMemo(() => [rehypeInlineCodeProperty, rehypeSuperscript, rehypeSubscript], [])

View File

@ -246,6 +246,28 @@ img.fullScreen {
font-size: .85rem; font-size: .85rem;
} }
/* Utility classes used in rehype plugins in md.js */
.subscript {
vertical-align: sub;
font-size: smaller;
}
.superscript {
vertical-align: super;
font-size: smaller;
}
.videoWrapper {
max-width: 320px;
padding-right: 15px;
margin: '0.5rem 0',
}
.videoWrapper.topLevel {
max-width: 640px;
}
.videoContainer { .videoContainer {
position: relative; position: relative;
width: 100%; width: 100%;
@ -262,14 +284,108 @@ img.fullScreen {
left: 0; left: 0;
} }
.twitterContainer, .nostrContainer {
/* Utility classes used in rehype plugins in md.js */ margin-top: .25rem;
.subscript { position: relative;
vertical-align: sub; overflow: hidden;
font-size: smaller;
} }
.superscript { .twitterContainer:not(:first-child), .nostrContainer:not(:first-child) {
vertical-align: super; margin-top: .75rem;
font-size: smaller; }
.twitterContainer iframe, .nostrContainer iframe {
border-radius: 13px;
}
.nostrContainer {
margin-right: 15px;
}
.videoWrapper:not(:first-child) {
margin-top: .75rem;
}
.twitterContained {
height: 150px;
overflow: hidden;
}
.twitterContained.topLevel {
height: 200px;
}
.twitterShowFull {
position: absolute;
bottom: 0;
left: 0;
border-radius: 0;
line-height: 1.2;
padding: .75rem 1rem;
}
.tweetsSkeleton {
display: flex;
flex-flow: row wrap;
max-width: 350px;
width: 100%;
padding-right: 12px;
}
.tweetsSkeleton.topLevel {
max-width: 550px;
}
.tweetSkeleton {
width: 100%;
}
.tweetSkeleton {
border: 0.05rem solid var(--theme-borderColor);
border-radius: 12px;
height: 150px;
padding: 1.5rem;
}
.topLevel .tweetSkeleton {
height: 200px;
}
.tweetSkeleton .img {
height: 48px;
width: 48px;
border-radius: 50%;
}
.tweetSkeleton .content1,
.tweetSkeleton .content2 {
height: 50%;
margin-top: 1rem;
}
.tweetSkeleton .line {
height: 25%;
margin: 0.5rem 0;
width: 100%;
border-radius: .4rem;
}
.tweetSkeleton .line:last-child {
width: 75%;
}
.wavlakeWrapper {
width: 100%;
height: 380px;
max-width: 480px;
border-radius: 1.65rem;
overflow: hidden;
}
.spotifyWrapper {
width: 100%;
height: 152px;
max-width: 480px;
border-radius: 13px;
overflow: hidden;
} }

View File

@ -10,7 +10,7 @@ export const defaultCommentSort = (pinned, bio, createdAt) => {
return 'hot' return 'hot'
} }
export const isJob = item => typeof item.maxBid !== 'undefined' export const isJob = item => item.maxBid !== null && typeof item.maxBid !== 'undefined'
// a delete directive preceded by a non word character that isn't a backtick // a delete directive preceded by a non word character that isn't a backtick
const deletePattern = /\B@delete\s+in\s+(\d+)\s+(second|minute|hour|day|week|month|year)s?/gi const deletePattern = /\B@delete\s+in\s+(\d+)\s+(second|minute|hour|day|week|month|year)s?/gi

View File

@ -1,3 +1,6 @@
import { nip19 } from 'nostr-tools'
import { DEFAULT_CROSSPOSTING_RELAYS } from './nostr'
export function ensureProtocol (value) { export function ensureProtocol (value) {
if (!value) return value if (!value) return value
value = value.trim() value = value.trim()
@ -76,9 +79,52 @@ export function parseInternalLinks (href) {
} }
export function parseEmbedUrl (href) { export function parseEmbedUrl (href) {
if (!href) return null
try { try {
const twitter = href.match(/^https?:\/\/(?:twitter|x)\.com\/(?:#!\/)?\w+\/status(?:es)?\/(?<id>\d+)/)
if (twitter?.groups?.id) {
return {
provider: 'twitter',
id: twitter.groups.id
}
}
const { hostname, pathname, searchParams } = new URL(href) const { hostname, pathname, searchParams } = new URL(href)
// nostr prefixes: [npub1, nevent1, nprofile1, note1]
const nostr = href.match(/(?<id>(?<type>npub1|nevent1|nprofile1|note1|naddr1)[02-9ac-hj-np-z]+)/)
if (nostr?.groups?.id) {
let id = nostr.groups.id
if (nostr.groups.type === 'npub1') {
const { data } = nip19.decode(id)
id = nip19.nprofileEncode({ pubkey: data })
}
if (nostr.groups.type === 'note1') {
const { data } = nip19.decode(id)
// njump needs relays to embed
id = nip19.neventEncode({ id: data, relays: DEFAULT_CROSSPOSTING_RELAYS, author: '' })
}
return {
provider: 'nostr',
id
}
}
// https://wavlake.com/track/c0aaeff8-5a26-49cf-8dad-2b6909e4aed1
if (hostname.endsWith('wavlake.com') && pathname.startsWith('/track')) {
return {
provider: 'wavlake',
id: pathname.split('/')?.[2]
}
}
if (hostname.endsWith('spotify.com') && pathname.startsWith('/track')) {
return {
provider: 'spotify'
}
}
if (hostname.endsWith('youtube.com')) { if (hostname.endsWith('youtube.com')) {
if (pathname.includes('/watch')) { if (pathname.includes('/watch')) {
return { return {
@ -130,12 +176,11 @@ export function parseEmbedUrl (href) {
} }
} }
} }
} catch { } catch (err) {
// ignore console.error('Error parsing embed URL:', err)
} }
// Important to return empty object as default return null
return {}
} }
export function stripTrailingSlash (uri) { export function stripTrailingSlash (uri) {

View File

@ -95,7 +95,7 @@ export function middleware (request) {
// unsafe-inline for styles is not ideal but okay if script-src is using nonces // unsafe-inline for styles is not ideal but okay if script-src is using nonces
"style-src 'self' a.stacker.news 'unsafe-inline'", "style-src 'self' a.stacker.news 'unsafe-inline'",
"manifest-src 'self'", "manifest-src 'self'",
'frame-src www.youtube.com platform.twitter.com rumble.com bitcointv.com peertube.tv', 'frame-src www.youtube.com platform.twitter.com njump.me open.spotify.com rumble.com embed.wavlake.com bitcointv.com peertube.tv',
"connect-src 'self' https: wss:" + devSrc, "connect-src 'self' https: wss:" + devSrc,
// disable dangerous plugins like Flash // disable dangerous plugins like Flash
"object-src 'none'", "object-src 'none'",

View File

@ -448,12 +448,12 @@ export default function Settings ({ ssrData }) {
/>} />}
<Checkbox <Checkbox
label={ label={
<div className='d-flex align-items-center'>only load images from proxy <div className='d-flex align-items-center'>do not load images, videos, or content from external sites
<Info> <Info>
<ul className='fw-bold'> <ul>
<li>only load images from our image proxy automatically</li> <li>only load images and videos when we can proxy them</li>
<li>this prevents IP address leaks to arbitrary sites</li> <li>this prevents IP address leaks to arbitrary sites</li>
<li>if we fail to load an image, the raw link will be shown</li> <li>if we can't, the raw link will be shown instead</li>
</ul> </ul>
</Info> </Info>
</div> </div>
@ -511,9 +511,22 @@ export default function Settings ({ ssrData }) {
/> />
<Checkbox <Checkbox
label={ label={
<div className='d-flex align-items-center'>show images and video <div className='d-flex align-items-center'>show images, video, and 3rd party embeds
<Info> <Info>
<p>disable to show images and videos as links instead of embedding them</p> <ul>
<li>if checked and a link is an image, video or can be embedded in another way, we will do it</li>
<li>we support embeds from following sites:</li>
<ul>
<li>njump.me</li>
<li>youtube.com</li>
<li>twitter.com</li>
<li>spotify.com</li>
<li>rumble.com</li>
<li>wavlake.com</li>
<li>bitcointv.com</li>
<li>peertube.tv</li>
</ul>
</ul>
</Info> </Info>
</div> </div>
} }

View File

@ -15,98 +15,6 @@
} }
} }
.videoWrapper {
max-width: 640px;
padding-right: 15px;
}
.videoContainer {
position: relative;
width: 100%;
height: 0;
padding-bottom: 56.25%;
overflow: hidden;
}
.videoContainer iframe {
width: 100%;
height: 100%;
position: absolute;
top: 0;
left: 0;
}
.twitterContainer {
margin-top: .25rem;
position: relative;
}
.twitterContainer:not(:first-child) {
margin-top: .75rem;
}
.videoWrapper:not(:first-child) {
margin-top: .75rem;
}
.twitterContainer iframe {
border-radius: 12px;
}
.twitterContained {
height: 200px;
overflow: hidden;
}
.twitterShowFull {
position: absolute;
bottom: 0;
left: 0;
border-radius: 0;
}
.tweetsSkeleton {
display: flex;
flex-flow: row wrap;
max-width: 550px;
width: 100%;
padding-right: 12px;
}
.tweetSkeleton {
width: 100%;
}
.tweetSkeleton {
border: 0.05rem solid var(--theme-borderColor);
border-radius: 12px;
height: 200px;
padding: 1.5rem;
}
.tweetSkeleton .img {
height: 48px;
width: 48px;
border-radius: 50%;
}
.tweetSkeleton .content1,
.tweetSkeleton .content2 {
height: 50%;
margin-top: 1rem;
}
.tweetSkeleton .line {
height: 25%;
margin: 0.5rem 0;
width: 100%;
border-radius: .4rem;
}
.tweetSkeleton .line:last-child {
width: 75%;
}
.fullItemContainer { .fullItemContainer {
margin-bottom: .5rem; margin-bottom: .5rem;
} }

1
svgs/video-on-fill.svg Normal file
View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="currentColor"><path d="M17 9.2L22.2133 5.55071C22.4395 5.39235 22.7513 5.44737 22.9096 5.6736C22.9684 5.75764 23 5.85774 23 5.96033V18.0397C23 18.3158 22.7761 18.5397 22.5 18.5397C22.3974 18.5397 22.2973 18.5081 22.2133 18.4493L17 14.8V19C17 19.5523 16.5523 20 16 20H2C1.44772 20 1 19.5523 1 19V5C1 4.44772 1.44772 4 2 4H16C16.5523 4 17 4.44772 17 5V9.2Z"></path></svg>

After

Width:  |  Height:  |  Size: 435 B