Embedding a <Player> into an <iframe>
info
Credit to Marcus Stenbeck for creating this snippet.
This snippet is useful if you want to isolate the global styles of your homepage from the global styles of the <Player>
, for example if you are using TailwindCSS.
Usagediff
- import { Player } from '@remotion/player';+ import { IframePlayer } from 'path/to/IframePlayer';- <Player {/* ... */} />+ <IframePlayer {/* ... */} />
Usagediff
- import { Player } from '@remotion/player';+ import { IframePlayer } from 'path/to/IframePlayer';- <Player {/* ... */} />+ <IframePlayer {/* ... */} />
IframePlayer.tsxtsx
import { Player } from '@remotion/player';import React, { ComponentProps, useEffect, useRef, useState } from 'react';import ReactDOM from 'react-dom';const className = '__player';const borderNone: React.CSSProperties = {border: 'none',};export const IframedPlayer: React.FC<ComponentProps<typeof Player>> = (props) => {const [contentRef, setContentRef] = useState<HTMLIFrameElement | null>(null);const resizeObserverRef = useRef<ResizeObserver | null>(null);const playerRef = useRef<HTMLDivElement | null>(null);const mountNode = contentRef?.contentDocument?.body;useEffect(() => {if (!contentRef || !contentRef.contentDocument) return;// Remove margin and padding so player fits snuglycontentRef.contentDocument.body.style.margin = '0';contentRef.contentDocument.body.style.padding = '0';// When player div is resized also resize iframeresizeObserverRef.current = new ResizeObserver(([playerEntry]) => {const playerRect = playerEntry.contentRect;contentRef.width = String(playerRect.width);contentRef.height = String(playerRect.height);});// The remotion player elementconst playerElement = contentRef.contentDocument.querySelector('.' + className);if (!playerElement) {throw new Error('Player element not found. Add a "' +className +'" class to the <Player>.');}// Watch the player element for size changesresizeObserverRef.current.observe(playerElement as Element);return () => {// ContentRef changed: unobserve!(resizeObserverRef.current as ResizeObserver).unobserve(playerElement as Element);};}, [contentRef]);const combinedClassName = `${className} ${props.className ?? ''}`.trim();return (// eslint-disable-next-line @remotion/warn-native-media-tag<iframe ref={setContentRef} style={borderNone}>{mountNode &&ReactDOM.createPortal(<Player {...props} className={combinedClassName} />,mountNode)}</iframe>);};
IframePlayer.tsxtsx
import { Player } from '@remotion/player';import React, { ComponentProps, useEffect, useRef, useState } from 'react';import ReactDOM from 'react-dom';const className = '__player';const borderNone: React.CSSProperties = {border: 'none',};export const IframedPlayer: React.FC<ComponentProps<typeof Player>> = (props) => {const [contentRef, setContentRef] = useState<HTMLIFrameElement | null>(null);const resizeObserverRef = useRef<ResizeObserver | null>(null);const playerRef = useRef<HTMLDivElement | null>(null);const mountNode = contentRef?.contentDocument?.body;useEffect(() => {if (!contentRef || !contentRef.contentDocument) return;// Remove margin and padding so player fits snuglycontentRef.contentDocument.body.style.margin = '0';contentRef.contentDocument.body.style.padding = '0';// When player div is resized also resize iframeresizeObserverRef.current = new ResizeObserver(([playerEntry]) => {const playerRect = playerEntry.contentRect;contentRef.width = String(playerRect.width);contentRef.height = String(playerRect.height);});// The remotion player elementconst playerElement = contentRef.contentDocument.querySelector('.' + className);if (!playerElement) {throw new Error('Player element not found. Add a "' +className +'" class to the <Player>.');}// Watch the player element for size changesresizeObserverRef.current.observe(playerElement as Element);return () => {// ContentRef changed: unobserve!(resizeObserverRef.current as ResizeObserver).unobserve(playerElement as Element);};}, [contentRef]);const combinedClassName = `${className} ${props.className ?? ''}`.trim();return (// eslint-disable-next-line @remotion/warn-native-media-tag<iframe ref={setContentRef} style={borderNone}>{mountNode &&ReactDOM.createPortal(<Player {...props} className={combinedClassName} />,mountNode)}</iframe>);};