It's late and I'm likely missing something simple. In React with Remotion, I decided to extend the <Series.Sequence>
component (doc link) for a set of narrated sequences, automatically setting the sequence duration based on the narration audio file time plus any time padding before and after the audio starts/finishes. For this, I created a new component, returning a <Series.Sequence> component. However, when used (see Composition.tsx below), I get an error:
The <Series /> component only accepts a list of <Series.Sequence /> components as it's children, but got [object Object] instead
This is showing after compile in the web browser:
node_modules/remotion/dist/cjs/series/index.js:34
if (castedChild.type !== SeriesSequence) {
throw new TypeError(`The <Series /> component only accepts a list of <Series.Sequence /> components as it's children, but got ${castedChild} instead`);
}
I'm thinking I need to cast the type here, yet I'm missing where this should occur at this late hour. What's the best way to proceed?
NarratedSeriesSequence.tsx (returning Series.Sequence):
import React, {useCallback, useEffect, useState} from 'react';
import {staticFile, Series, useVideoConfig} from 'remotion';
import {getAudioDurationInSeconds} from '@remotion/media-utils';
export interface NarratedSeriesSequenceProps {
narrationAudioFile: string;
children: JSX.Element;
framePaddingBefore?: number;
framePaddingAfter?: number;
}
/**
* Time a sequence dynamically based on narration audio length. Add optional padding before
* and after the narration begins. This gives more flexibility and time savings when needing
* to make use of several narration-based sequences and when making revisions to
* narration recordings.
*/
export const NarratedSeriesSequence: React.FC<NarratedSeriesSequenceProps> = ({
narrationAudioFile,
framePaddingBefore = 0,
framePaddingAfter = 0,
children,
}: NarratedSeriesSequenceProps) => {
const {fps} = useVideoConfig();
const [durationFrames, setDurationFrames] = useState(0);
const getDurationFrames = useCallback(async () => {
try {
const duration = await getAudioDurationInSeconds(
staticFile(narrationAudioFile)
);
const durationInFrames =
duration * fps + framePaddingBefore + framePaddingAfter;
setDurationFrames(durationInFrames);
} catch (err) {
console.log(err);
}
}, [narrationAudioFile, framePaddingBefore, framePaddingAfter, fps]);
// Get narration audio frames
useEffect(() => {
getDurationFrames();
}, [getDurationFrames]);
if (durationFrames === 0) {
return null;
}
return (
<Series.Sequence durationInFrames={durationFrames}>
{/* narration audio placed here soon */}
{children}
</Series.Sequence>
);
};
Composition.tsx
import {AbsoluteFill, Series} from 'remotion';
import {Logo} from './Logo';
import {Subtitle} from './Subtitle';
import {Title} from './Title';
import {z} from 'zod';
import {zColor} from '@remotion/zod-types';
import {BackgroundAudio} from './BackgroundAudio';
import {NarratedSeriesSequence} from './NarratedSeriesSequence';
export const myCompSchema = z.object({
titleText: z.string(),
titleColor: zColor(),
logoColor: zColor(),
});
export const MyComposition: React.FC<z.infer<typeof myCompSchema>> = ({
titleText: propOne,
titleColor: propTwo,
logoColor: propThree,
}) => {
return (
<>
<BackgroundAudio />
<Series>
{/* Series.Sequence: works fine if I remove NarratedSeriesSequence lines */}
<Series.Sequence durationInFrames={220}>
<AbsoluteFill className="bg-gray-100 items-center justify-center">
<div className="m-10" />
<Logo logoColor={propThree} />
<div className="m-3" />
<Title titleText={propOne} titleColor={propTwo} />
<Subtitle />
</AbsoluteFill>
</Series.Sequence>
{/* NarratedSeriesSequence returns a Series.Sequence, yet I may not be defining this correctly? Thinks [object Object] */}
<NarratedSeriesSequence narrationAudioFile="audio/narration/00-this-video-was.mp3">
<div>
This video was designed and rendered using React and Remotion.
</div>
</NarratedSeriesSequence>
{/* NarratedSeriesSequence returns a Series.Sequence, yet I may not be defining this correctly? */}
<NarratedSeriesSequence narrationAudioFile="audio/narration/01-remotion-gives-web-developers.mp3">
<div>
Remotion gives web developers and designers a set of libraries to
bring familiar tools into video projects, allowing for innovative
creativity that traditional video production tools handle very
differently.
</div>
</NarratedSeriesSequence>
</Series>
</>
);
};
The library does a reference equality check for the
Series.Sequence
function with the children, so it fails. Here's a dirty workaround -