.agents/skills/interactivity-best-practices/SKILL.md
Use these guidelines when creating or reviewing Remotion animations, especially if the animation should be editable in Remotion Studio Visual Mode.
interpolate() over standalone spring()Prefer interpolate() for most animation values.
interpolate() keyframes.When an animation should feel like a spring, keep the editable value in interpolate() and pass Easing.spring() as the easing function.
Prefer:
style={{
scale: interpolate(frame, [0, 30], [0, 1], {
easing: Easing.spring({
damping: 200,
}),
}),
}}
❌ Avoid using spring() as a separate driver when the same motion can be expressed as interpolate() easing:
const scale = spring({
frame,
fps,
config: {
damping: 200,
},
});
style={{
scale,
}}
Easing.spring() supports damping, mass, stiffness, and overshootClamping. It is normalized to the interpolation progress, so it does not take frame, fps, from, to, delay, reverse, durationInFrames, or durationRestThreshold.
Use standalone spring() only when the animation specifically needs a frame-driven physical spring simulation that cannot be represented as an easing on interpolate().
Use individual CSS transform properties such as scale, rotate, and translate instead of composing a transform string.
Prefer:
style={{
scale: interpolate(frame, [0, 100], [0, 2]),
}}
❌ Avoid:
style={{
transform: `scale(${interpolate(frame, [0, 100], [0, 2])})`,
}}
Individual transform properties are editable in Studio Visual Mode. Values hidden inside a transform string are not.
Put the interpolation directly in the JSX style object when the value should be visually editable.
Prefer:
style={{
scale: interpolate(frame, [0, 100], [0, 2]),
}}
❌ Avoid:
const scale = interpolate(frame, [0, 100], [0, 2]);
style={{
scale,
}}
Inline computations and literal values let Studio Visual Mode discover and edit the keyframes and props.
This also applies to effect parameters. Keep editable effect values inline in the effect call.
Prefer:
effects={[
radialProgressiveBlur({
center: [0.5, 0.5],
point1: [0.86, 0.36],
point2: [0.68, 0.84],
}),
]}
❌ Avoid hiding editable values behind constants:
const center = [0.5, 0.5] as const;
effects={[
radialProgressiveBlur({
center,
}),
]}
When using the effects prop, keep the effect array shape unconditional. Studio Visual Mode cannot reliably edit an effect that only exists behind a conditional expression.
Prefer rendering separate elements when one version should have effects and another should not:
<CanvasImage src={src} width={1280} height={720} />
<CanvasImage
src={src}
width={1280}
height={720}
effects={[
blur({
radius: interpolate(frame, [0, 100], [0, 40]),
}),
]}
/>
❌ Avoid conditionally including an effect:
<CanvasImage
src={src}
width={1280}
height={720}
effects={enabled ? [blur({radius: 40})] : []}
/>
Keep editable effect parameters inline inside the unconditional effect call. Do not pass variables for editable values such as coordinates, colors, numeric controls, or interpolations.
Interpolate CSS unit strings directly for rotate and translate.
Prefer:
style={{
rotate: interpolate(frame, [0, 100], ["20deg", "90deg"]),
translate: interpolate(frame, [0, 100], ["0px 0px", "100px 100px"]),
}}
❌ Avoid manually building strings around separately interpolated numbers when the property can be interpolated directly.
Direct interpolation of these properties is supported and works better with visual editing in Studio.
Interactive.* for tweakable elementsUse the Interactive namespace from remotion when an element is likely to be tweaked in Studio.
import {Interactive} from "remotion";
For example:
<Interactive.Div
style={{
color: "red",
fontSize: 80,
}}
>
Hello
</Interactive.Div>
<Interactive.Div>, <Interactive.Span>, and the other Interactive.* components behave like their regular HTML equivalents, but enable interactivity in Studio.
Inline styles on these elements, such as text color and font size, can be standardized interactively.