UNPKG

remotion

Version:

Make videos programmatically

509 lines (508 loc) • 23.3 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); Object.defineProperty(exports, "__esModule", { value: true }); exports.useSharedAudio = exports.SharedAudioTagsContextProvider = exports.SharedAudioContextProvider = exports.SharedAudioTagsContext = exports.SharedAudioContext = void 0; const jsx_runtime_1 = require("react/jsx-runtime"); const react_1 = __importStar(require("react")); const log_level_context_js_1 = require("../log-level-context.js"); const log_js_1 = require("../log.js"); const play_and_handle_not_allowed_error_js_1 = require("../play-and-handle-not-allowed-error.js"); const use_remotion_environment_js_1 = require("../use-remotion-environment.js"); const shared_element_source_node_js_1 = require("./shared-element-source-node.js"); const use_audio_context_js_1 = require("./use-audio-context.js"); const wait_until_actually_resumed_js_1 = require("./wait-until-actually-resumed.js"); const EMPTY_AUDIO = 'data:audio/mp3;base64,/+MYxAAJcAV8AAgAABn//////+/gQ5BAMA+D4Pg+BAQBAEAwD4Pg+D4EBAEAQDAPg++hYBH///hUFQVBUFREDQNHmf///////+MYxBUGkAGIMAAAAP/29Xt6lUxBTUUzLjEwMFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVV/+MYxDUAAANIAAAAAFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVV'; const compareProps = (obj1, obj2) => { const keysA = Object.keys(obj1).sort(); const keysB = Object.keys(obj2).sort(); if (keysA.length !== keysB.length) { return false; } for (let i = 0; i < keysA.length; i++) { // Not the same keys if (keysA[i] !== keysB[i]) { return false; } // Not the same values if (obj1[keysA[i]] !== obj2[keysB[i]]) { return false; } } return true; }; const didPropChange = (key, newProp, prevProp) => { // /music.mp3 and http://localhost:3000/music.mp3 are the same if (key === 'src' && !prevProp.startsWith('data:') && !newProp.startsWith('data:')) { return (new URL(prevProp, window.origin).toString() !== new URL(newProp, window.origin).toString()); } if (prevProp === newProp) { return false; } return true; }; exports.SharedAudioContext = (0, react_1.createContext)(null); exports.SharedAudioTagsContext = (0, react_1.createContext)(null); const shouldSaveForLater = (state) => { if (state === 'suspended' || state === 'running-to-suspended' || state === 'interrupted') { return true; } if (state === 'running' || state === 'suspended-to-running') { return false; } throw new Error(`Unexpected audio context state: ${state}`); }; const SharedAudioContextProvider = ({ children, audioLatencyHint, audioEnabled }) => { const logLevel = (0, log_level_context_js_1.useLogLevel)(); const ctxAndGain = (0, use_audio_context_js_1.useSingletonAudioContext)({ logLevel, latencyHint: audioLatencyHint, audioEnabled, }); const audioContextIsPlayingEventually = (0, react_1.useRef)(false); const isResuming = (0, react_1.useRef)(null); const audioSyncAnchor = (0, react_1.useMemo)(() => ({ value: 0 }), []); const audioSyncAnchorListeners = (0, react_1.useRef)([]); const audioSyncAnchorEmitter = (0, react_1.useMemo)(() => { return { dispatch: (event) => { audioSyncAnchorListeners.current.forEach((l) => l(event)); }, subscribe: (listener) => { audioSyncAnchorListeners.current.push(listener); return { remove: () => { audioSyncAnchorListeners.current = audioSyncAnchorListeners.current.filter((l) => l !== listener); }, }; }, }; }, []); const prevEndTimes = (0, react_1.useRef)({ scheduledEndTime: null, mediaEndTime: null }); const nodesToResume = (0, react_1.useRef)(new Map()); const unscheduleAudioNode = (0, react_1.useCallback)((node) => { nodesToResume.current.delete(node); }, []); const scheduleAudioNode = (0, react_1.useMemo)(() => { return ({ node, mediaTimestamp, scheduledTime, duration, offset, originalUnloopedMediaTimestamp, }) => { if (!ctxAndGain) { throw new Error('Audio context not found'); } const currentState = ctxAndGain.getState(); if (currentState === 'closed') { return { type: 'not-started', reason: 'audio context is closed', }; } const saveForLater = shouldSaveForLater(currentState); if (duration > 0) { if (saveForLater) { nodesToResume.current.set(node, { scheduledTime, offset, duration, }); } else { node.start(scheduledTime, offset, duration); } } const scheduledEndTime = scheduledTime + duration / node.playbackRate.value; const mediaTime = mediaTimestamp + offset; const mediaEndTime = mediaTime + duration; const latency = ctxAndGain.audioContext.baseLatency + ctxAndGain.audioContext.outputLatency; const timeDiff = scheduledTime - ctxAndGain.audioContext.currentTime; const prev = prevEndTimes.current; const scheduledMismatch = prev.scheduledEndTime !== null && Math.abs(scheduledTime - prev.scheduledEndTime) > 0.001; const mediaMismatch = prev.mediaEndTime !== null && Math.abs(mediaTime - prev.mediaEndTime) > 0.001; log_js_1.Log.verbose({ logLevel, tag: 'audio-scheduling' }, 'scheduled %c%s%c %s %c%s%c %s %c%s%c %s %s %s %s %s', scheduledMismatch ? 'color: red; font-weight: bold' : '', scheduledTime.toFixed(4), '', scheduledEndTime.toFixed(4), mediaMismatch ? 'color: red; font-weight: bold' : '', mediaTime.toFixed(4), '', mediaEndTime.toFixed(4), duration < 0 ? 'color: red; font-weight: bold' : timeDiff < 0 ? 'color: red; font-weight: bold' : 'color: blue; font-weight: bold', duration < 0 ? 'missed ' + Math.abs(offset).toFixed(2) + 's' : Math.abs(timeDiff).toFixed(2) + (timeDiff < 0 ? ' delay' : ' ahead'), '', 'current=' + ctxAndGain.audioContext.currentTime.toFixed(4), 'offset=' + offset.toFixed(4), 'latency=' + latency.toFixed(4), 'state=' + ctxAndGain.audioContext.state, originalUnloopedMediaTimestamp !== mediaTime ? 'original_ts=' + originalUnloopedMediaTimestamp.toFixed(4) : '', 'action=' + (saveForLater ? 'schedule' : 'start'), ''); prev.scheduledEndTime = scheduledEndTime; prev.mediaEndTime = mediaEndTime; return duration > 0 ? { type: 'started', scheduledTime, } : { type: 'not-started', reason: 'missed ' + Math.abs(offset).toFixed(2) + 's', }; }; }, [ctxAndGain, logLevel]); const resume = (0, react_1.useCallback)(() => { if (!ctxAndGain) { return Promise.resolve(); } if (audioContextIsPlayingEventually.current) { return Promise.resolve(); } audioContextIsPlayingEventually.current = true; ctxAndGain.gainNode.gain.cancelScheduledValues(ctxAndGain.audioContext.currentTime); ctxAndGain.gainNode.gain.setValueAtTime(0, ctxAndGain.audioContext.currentTime); ctxAndGain.gainNode.gain.linearRampToValueAtTime(1, ctxAndGain.audioContext.currentTime + 0.03); nodesToResume.current.forEach((r, node) => { node.start(r.scheduledTime, r.offset, r.duration); }); nodesToResume.current.clear(); const resumePromise = ctxAndGain.resume(); isResuming.current = new Promise((resolve) => { (0, wait_until_actually_resumed_js_1.waitUntilActuallyResumed)(ctxAndGain.audioContext, logLevel).then(resolve); resumePromise.catch((err) => { log_js_1.Log.warn({ logLevel, tag: 'audio' }, 'AudioContext resume rejected, continuing without audio sync', err); resolve(); }); }).finally(() => { isResuming.current = null; }); return resumePromise.catch(() => { // Already logged above; swallow to avoid unhandled rejection // since callers (e.g. use-playback.ts) do not await this. }); }, [ctxAndGain, logLevel]); const getIsResumingAudioContext = (0, react_1.useCallback)(() => { return isResuming.current; }, []); const suspend = (0, react_1.useCallback)(() => { if (!ctxAndGain) { return Promise.resolve(); } if (!audioContextIsPlayingEventually.current) { return Promise.resolve(); } audioContextIsPlayingEventually.current = false; return ctxAndGain.suspend(); }, [ctxAndGain]); const audioContextValue = (0, react_1.useMemo)(() => { var _a, _b; return { audioContext: (_a = ctxAndGain === null || ctxAndGain === void 0 ? void 0 : ctxAndGain.audioContext) !== null && _a !== void 0 ? _a : null, getAudioContextState: () => { var _a; return (_a = ctxAndGain === null || ctxAndGain === void 0 ? void 0 : ctxAndGain.getState()) !== null && _a !== void 0 ? _a : null; }, gainNode: (_b = ctxAndGain === null || ctxAndGain === void 0 ? void 0 : ctxAndGain.gainNode) !== null && _b !== void 0 ? _b : null, audioSyncAnchor, audioSyncAnchorEmitter, scheduleAudioNode, resume, suspend, getIsResumingAudioContext, unscheduleAudioNode, }; }, [ ctxAndGain, audioSyncAnchor, audioSyncAnchorEmitter, scheduleAudioNode, resume, suspend, getIsResumingAudioContext, unscheduleAudioNode, ]); return ((0, jsx_runtime_1.jsx)(exports.SharedAudioContext.Provider, { value: audioContextValue, children: children })); }; exports.SharedAudioContextProvider = SharedAudioContextProvider; const SharedAudioTagsContextProvider = ({ children, numberOfAudioTags }) => { var _a, _b; const audios = (0, react_1.useRef)([]); const [initialNumberOfAudioTags] = (0, react_1.useState)(numberOfAudioTags); if (numberOfAudioTags !== initialNumberOfAudioTags) { throw new Error('The number of shared audio tags has changed dynamically. Once you have set this property, you cannot change it afterwards.'); } const logLevel = (0, log_level_context_js_1.useLogLevel)(); const mountTime = (0, log_level_context_js_1.useMountTime)(); const env = (0, use_remotion_environment_js_1.useRemotionEnvironment)(); const audioCtx = (0, react_1.useContext)(exports.SharedAudioContext); const audioContext = (_a = audioCtx === null || audioCtx === void 0 ? void 0 : audioCtx.audioContext) !== null && _a !== void 0 ? _a : null; const resume = audioCtx === null || audioCtx === void 0 ? void 0 : audioCtx.resume; const refs = (0, react_1.useMemo)(() => { return new Array(numberOfAudioTags).fill(true).map(() => { const ref = (0, react_1.createRef)(); return { id: Math.random(), ref, mediaElementSourceNode: audioContext ? (0, shared_element_source_node_js_1.makeSharedElementSourceNode)({ audioContext, ref, }) : null, }; }); }, [audioContext, numberOfAudioTags]); /** * Effects in React 18 fire twice, and we are looking for a way to only fire it once. * - useInsertionEffect only fires once. If it's available we are in React 18. * - useLayoutEffect only fires once in React 17. * * Need to import it from React to fix React 17 ESM support. */ const effectToUse = (_b = react_1.default.useInsertionEffect) !== null && _b !== void 0 ? _b : react_1.default.useLayoutEffect; // Disconnecting the SharedElementSourceNodes if the Player unmounts to prevent leak. // https://github.com/remotion-dev/remotion/issues/6285 // But useInsertionEffect will fire before other effects, meaning the // nodes might still be used. Using rAF to ensure it's after other effects. effectToUse(() => { return () => { requestAnimationFrame(() => { refs.forEach(({ mediaElementSourceNode }) => { mediaElementSourceNode === null || mediaElementSourceNode === void 0 ? void 0 : mediaElementSourceNode.cleanup(); }); }); }; }, [refs]); const takenAudios = (0, react_1.useRef)(new Array(numberOfAudioTags).fill(false)); const rerenderAudios = (0, react_1.useCallback)(() => { refs.forEach(({ ref, id }) => { var _a; const data = (_a = audios.current) === null || _a === void 0 ? void 0 : _a.find((a) => a.id === id); const { current } = ref; if (!current) { // Whole player has been unmounted, the refs don't exist anymore. // It is not an error anymore though return; } if (data === undefined) { current.src = EMPTY_AUDIO; return; } if (!data) { throw new TypeError('Expected audio data to be there'); } Object.keys(data.props).forEach((key) => { // @ts-expect-error if (didPropChange(key, data.props[key], current[key])) { // @ts-expect-error current[key] = data.props[key]; } }); }); }, [refs]); const registerAudio = (0, react_1.useCallback)((options) => { var _a, _b; const { aud, audioId, premounting, postmounting } = options; const found = (_a = audios.current) === null || _a === void 0 ? void 0 : _a.find((a) => a.audioId === audioId); if (found) { return found; } const firstFreeAudio = takenAudios.current.findIndex((a) => a === false); if (firstFreeAudio === -1) { throw new Error(`Tried to simultaneously mount ${numberOfAudioTags + 1} <Html5Audio /> tags at the same time. With the current settings, the maximum amount of <Html5Audio /> tags is limited to ${numberOfAudioTags} at the same time. Remotion pre-mounts silent audio tags to help avoid browser autoplay restrictions. See https://remotion.dev/docs/player/autoplay#using-the-numberofsharedaudiotags-prop for more information on how to increase this limit.`); } const { id, ref, mediaElementSourceNode } = refs[firstFreeAudio]; const cloned = [...takenAudios.current]; cloned[firstFreeAudio] = id; takenAudios.current = cloned; const newElem = { props: aud, id, el: ref, audioId, mediaElementSourceNode, premounting, audioMounted: Boolean(ref.current), postmounting, cleanupOnMediaTagUnmount: () => { // Don't disconnect here, only when the Player unmounts. }, }; (_b = audios.current) === null || _b === void 0 ? void 0 : _b.push(newElem); rerenderAudios(); return newElem; }, [numberOfAudioTags, refs, rerenderAudios]); const unregisterAudio = (0, react_1.useCallback)((id) => { var _a; const cloned = [...takenAudios.current]; const index = refs.findIndex((r) => r.id === id); if (index === -1) { throw new TypeError('Error occured in '); } cloned[index] = false; takenAudios.current = cloned; audios.current = (_a = audios.current) === null || _a === void 0 ? void 0 : _a.filter((a) => a.id !== id); rerenderAudios(); }, [refs, rerenderAudios]); const updateAudio = (0, react_1.useCallback)(({ aud, audioId, id, premounting, postmounting, }) => { var _a; let changed = false; audios.current = (_a = audios.current) === null || _a === void 0 ? void 0 : _a.map((prevA) => { const audioMounted = Boolean(prevA.el.current); if (prevA.audioMounted !== audioMounted) { changed = true; } if (prevA.id === id) { const isTheSame = compareProps(aud, prevA.props) && prevA.premounting === premounting && prevA.postmounting === postmounting; if (isTheSame) { return prevA; } changed = true; return { ...prevA, props: aud, premounting, postmounting, audioId, audioMounted, }; } return prevA; }); if (changed) { rerenderAudios(); } }, [rerenderAudios]); const playAllAudios = (0, react_1.useCallback)(() => { refs.forEach((ref) => { const audio = audios.current.find((a) => a.el === ref.ref); if (audio === null || audio === void 0 ? void 0 : audio.premounting) { return; } (0, play_and_handle_not_allowed_error_js_1.playAndHandleNotAllowedError)({ mediaRef: ref.ref, mediaType: 'audio', onAutoPlayError: null, logLevel, mountTime, reason: 'playing all audios', isPlayer: env.isPlayer, }); }); resume === null || resume === void 0 ? void 0 : resume(); }, [logLevel, mountTime, refs, env.isPlayer, resume]); const audioTagsValue = (0, react_1.useMemo)(() => { return { registerAudio, unregisterAudio, updateAudio, playAllAudios, numberOfAudioTags, }; }, [ numberOfAudioTags, playAllAudios, registerAudio, unregisterAudio, updateAudio, ]); return ((0, jsx_runtime_1.jsxs)(exports.SharedAudioTagsContext.Provider, { value: audioTagsValue, children: [refs.map(({ id, ref }) => { return ( // Without preload="metadata", iOS will seek the time internally // but not actually with sound. Adding `preload="metadata"` helps here. // https://discord.com/channels/809501355504959528/817306414069710848/1130519583367888906 (0, jsx_runtime_1.jsx)("audio", { ref: ref, preload: "metadata", src: EMPTY_AUDIO }, id)); }), children] })); }; exports.SharedAudioTagsContextProvider = SharedAudioTagsContextProvider; const useSharedAudio = ({ aud, audioId, premounting, postmounting, }) => { var _a; const audioCtx = (0, react_1.useContext)(exports.SharedAudioContext); const tagsCtx = (0, react_1.useContext)(exports.SharedAudioTagsContext); /** * We work around this in React 18 so an audio tag will only register itself once */ const [elem] = (0, react_1.useState)(() => { if (tagsCtx && tagsCtx.numberOfAudioTags > 0) { return tagsCtx.registerAudio({ aud, audioId, premounting, postmounting }); } // numberOfSharedAudioTags is 0 const el = react_1.default.createRef(); const mediaElementSourceNode = (audioCtx === null || audioCtx === void 0 ? void 0 : audioCtx.audioContext) ? (0, shared_element_source_node_js_1.makeSharedElementSourceNode)({ audioContext: audioCtx.audioContext, ref: el, }) : null; return { el, id: Math.random(), props: aud, audioId, mediaElementSourceNode, premounting, audioMounted: Boolean(el.current), postmounting, cleanupOnMediaTagUnmount: () => { mediaElementSourceNode === null || mediaElementSourceNode === void 0 ? void 0 : mediaElementSourceNode.cleanup(); }, }; }); /** * Effects in React 18 fire twice, and we are looking for a way to only fire it once. * - useInsertionEffect only fires once. If it's available we are in React 18. * - useLayoutEffect only fires once in React 17. * * Need to import it from React to fix React 17 ESM support. */ const effectToUse = (_a = react_1.default.useInsertionEffect) !== null && _a !== void 0 ? _a : react_1.default.useLayoutEffect; if (typeof document !== 'undefined') { effectToUse(() => { if (tagsCtx && tagsCtx.numberOfAudioTags > 0) { tagsCtx.updateAudio({ id: elem.id, aud, audioId, premounting, postmounting, }); } }, [aud, tagsCtx, elem.id, audioId, premounting, postmounting]); effectToUse(() => { return () => { if (tagsCtx && tagsCtx.numberOfAudioTags > 0) { tagsCtx.unregisterAudio(elem.id); } }; }, [tagsCtx, elem.id]); } return elem; }; exports.useSharedAudio = useSharedAudio;