Skip to content

Commit

Permalink
Embed Rumble Video (#1191)
Browse files Browse the repository at this point in the history
* Render Rumble video in preview and posts

* Display Rumble video

* Remove workspace

* Add util function

* Use searchParam for id

* Update check for Rumble

* Update youtube match strings

* fix hostname conditions

---------

Co-authored-by: keyan <[email protected]>
  • Loading branch information
tsmith123 and huumn authored May 28, 2024
1 parent 9c5bec0 commit 52f57f8
Show file tree
Hide file tree
Showing 6 changed files with 101 additions and 24 deletions.
28 changes: 22 additions & 6 deletions components/item-full.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import Share from './share'
import Toc from './table-of-contents'
import Link from 'next/link'
import { RootProvider } from './root'
import { IMGPROXY_URL_REGEXP } from '@/lib/url'
import { IMGPROXY_URL_REGEXP, parseEmbedUrl } from '@/lib/url'
import { numWithUnits } from '@/lib/format'
import { useQuoteReply } from './use-quote-reply'
import { UNKNOWN_LINK_REL } from '@/lib/constants'
Expand Down Expand Up @@ -70,6 +70,7 @@ function ItemEmbed ({ item }) {
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 (
Expand All @@ -83,21 +84,36 @@ function ItemEmbed ({ item }) {
)
}

const youtube = item.url?.match(/(https?:\/\/)?((www\.)?(youtube(-nocookie)?|youtube.googleapis)\.com.*(v\/|v=|vi=|vi\/|e\/|embed\/|user\/.*\/u\/\d+\/)|youtu\.be\/)(?<id>[_0-9a-z-]+)((?:\?|&)(?:t|start)=(?<start>\d+))?/i)
if (youtube?.groups?.id) {
const { provider, id, meta } = parseEmbedUrl(item.url)

if (provider === 'youtube') {
return (
<div className={styles.youtubeContainerContainer}>
<div className={styles.videoWrapper}>
<YouTube
videoId={youtube.groups.id} className={styles.youtubeContainer} opts={{
videoId={id} className={styles.videoContainer} opts={{
playerVars: {
start: youtube?.groups?.start
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}
/>
</div>
</div>
)
}

if (item.url?.match(IMGPROXY_URL_REGEXP)) {
return <ZoomableImage src={item.url} rel={item.rel ?? UNKNOWN_LINK_REL} />
}
Expand Down
36 changes: 29 additions & 7 deletions components/text.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import Thumb from '@/svgs/thumb-up-fill.svg'
import { toString } from 'mdast-util-to-string'
import copy from 'clipboard-copy'
import ZoomableImage, { decodeOriginalUrl } from './image'
import { IMGPROXY_URL_REGEXP, parseInternalLinks } from '@/lib/url'
import { IMGPROXY_URL_REGEXP, parseInternalLinks, parseEmbedUrl } from '@/lib/url'
import reactStringReplace from 'react-string-replace'
import { rehypeInlineCodeProperty } from '@/lib/md'
import { Button } from 'react-bootstrap'
Expand Down Expand Up @@ -238,22 +238,44 @@ export default memo(function Text ({ rel, imgproxyUrls, children, tab, itemId, o
// ignore errors like invalid URLs
}

// if the link is to a youtube video, render the video
const youtube = href.match(/(https?:\/\/)?((www\.)?(youtube(-nocookie)?|youtube.googleapis)\.com.*(v\/|v=|vi=|vi\/|e\/|embed\/|user\/.*\/u\/\d+\/)|youtu\.be\/)(?<id>[_0-9a-z-]+)((?:\?|&)(?:t|start)=(?<start>\d+))?/i)
if (youtube?.groups?.id) {
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={{ maxWidth: topLevel ? '640px' : '320px', paddingRight: '15px', margin: '0.5rem 0' }}>
<div style={videoWrapperStyles}>
<YouTube
videoId={youtube.groups.id} className={styles.youtubeContainer} opts={{
videoId={id} className={styles.videoContainer} opts={{
playerVars: {
start: youtube?.groups?.start
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}
/>
</div>
</div>
)
}

// assume the link is an image which will fallback to link if it's not
return <Img src={href} rel={rel ?? UNKNOWN_LINK_REL} {...props}>{children}</Img>
},
Expand Down
4 changes: 2 additions & 2 deletions components/text.module.css
Original file line number Diff line number Diff line change
Expand Up @@ -237,15 +237,15 @@ img.fullScreen {
font-size: .85rem;
}

.youtubeContainer {
.videoContainer {
position: relative;
width: 100%;
height: 0;
padding-bottom: 56.25%;
overflow: hidden;
}

.youtubeContainer iframe {
.videoContainer iframe {
width: 100%;
height: 100%;
position: absolute;
Expand Down
39 changes: 39 additions & 0 deletions lib/url.js
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,45 @@ export function parseInternalLinks (href) {
}
}

export function parseEmbedUrl (href) {
const { hostname, pathname, searchParams } = new URL(href)

if (hostname.endsWith('youtube.com') && pathname.includes('/watch')) {
return {
provider: 'youtube',
id: searchParams.get('v'),
meta: {
href,
start: searchParams.get('t')
}
}
}

if (hostname.endsWith('youtu.be') && pathname.length > 1) {
return {
provider: 'youtube',
id: pathname.slice(1), // remove leading slash
meta: {
href,
start: searchParams.get('t')
}
}
}

if (hostname.endsWith('rumble.com') && pathname.includes('/embed')) {
return {
provider: 'rumble',
id: null, // not required
meta: {
href
}
}
}

// Important to return empty object as default
return {}
}

export function stripTrailingSlash (uri) {
return uri.endsWith('/') ? uri.slice(0, -1) : uri
}
Expand Down
2 changes: 1 addition & 1 deletion middleware.js
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ export function middleware (request) {
// unsafe-inline for styles is not ideal but okay if script-src is using nonces
"style-src 'self' a.stacker.news 'unsafe-inline'",
"manifest-src 'self'",
'frame-src www.youtube.com platform.twitter.com',
'frame-src www.youtube.com platform.twitter.com rumble.com',
"connect-src 'self' https: wss:" + devSrc,
// disable dangerous plugins like Flash
"object-src 'none'",
Expand Down
16 changes: 8 additions & 8 deletions styles/item.module.css
Original file line number Diff line number Diff line change
Expand Up @@ -15,15 +15,20 @@
}
}

.youtubeContainer {
.videoWrapper {
max-width: 640px;
padding-right: 15px;
}

.videoContainer {
position: relative;
width: 100%;
height: 0;
padding-bottom: 56.25%;
overflow: hidden;
}

.youtubeContainer iframe {
.videoContainer iframe {
width: 100%;
height: 100%;
position: absolute;
Expand All @@ -36,16 +41,11 @@
position: relative;
}

.youtubeContainerContainer {
max-width: 640px;
padding-right: 15px;
}

.twitterContainer:not(:first-child) {
margin-top: .75rem;
}

.youtubeContainerContainer:not(:first-child) {
.videoWrapper:not(:first-child) {
margin-top: .75rem;
}

Expand Down

0 comments on commit 52f57f8

Please sign in to comment.