design/motion-design/skills/timeline-choreographer/SKILL.md
This skill should be used when the user needs to orchestrate multiple elements animating in sequence or with complex timing relationships. Trigger when the user mentions "stagger animations", "animation sequence", "choreograph animations", "elements animate one after another", "staggered reveal", "delay between animations", "GSAP timeline", "Framer Motion variants", "animation timing", "orchestrate", "50ms stagger", or has a component where multiple elements need to animate in a specific order. Also trigger when the user says an animation "looks like everything explodes at once" or wants a more "coordinated" or "sequential" entrance.
npx skillsauth add harsh040506/claude-code-unified-skill-plugin-library timeline-choreographerInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
When multiple elements animate together, the difference between "everything flashed at once" and "beautifully sequenced" is choreography: the deliberate arrangement of when each element moves, for how long, and in what relationship to others.
Core principle: The eye follows the first thing that moves. Everything after should feel like a logical consequence.
Variants propagate from parent to children automatically. The parent's staggerChildren delays each child's entry sequentially.
const containerVariants = {
hidden: {},
visible: {
transition: {
staggerChildren: 0.06, // 60ms between each child
delayChildren: 0.1, // Initial delay before first child
when: "beforeChildren", // Parent transitions first, then children
},
},
};
const itemVariants = {
hidden: { opacity: 0, y: 16, filter: "blur(4px)" },
visible: {
opacity: 1,
y: 0,
filter: "blur(0px)",
transition: { duration: 0.3, ease: [0, 0, 0.58, 1] },
},
};
function StaggeredContent({ items }) {
return (
<motion.ul
variants={containerVariants}
initial="hidden"
animate="visible"
>
{items.map((item, i) => (
<motion.li key={i} variants={itemVariants}>
{item}
</motion.li>
))}
</motion.ul>
);
}
Dashboard grid stagger (diagonal pattern):
// Diagonal stagger — more interesting than row-by-row
const gridVariants = {
visible: {
transition: {
staggerChildren: 0.04,
delayChildren: 0.05,
},
},
};
// Items get delay based on their grid position for diagonal effect
const cardVariants = {
hidden: { opacity: 0, scale: 0.96 },
visible: (i) => ({
opacity: 1,
scale: 1,
transition: {
delay: (Math.floor(i / cols) + (i % cols)) * 0.04, // row + col diagonal
duration: 0.28,
},
}),
};
Use GSAP when:
import gsap from "gsap";
const tl = gsap.timeline({
defaults: { ease: "power2.out" },
delay: 0.1,
});
// Position parameter controls overlap:
// ">-0.3" = start 300ms before previous ends (overlap)
// "+=0.1" = start 100ms after previous ends (gap)
// "<" = start at same time as previous (concurrent)
tl.from(".hero-eyebrow", { y: 20, opacity: 0, duration: 0.4 })
.from(".hero-headline", { y: 30, opacity: 0, duration: 0.5 }, "-=0.2")
.from(".hero-subhead", { y: 20, opacity: 0, duration: 0.4 }, "-=0.2")
.from(".hero-cta", { y: 15, opacity: 0, duration: 0.35 }, "-=0.15")
.from(".hero-image", { scale: 0.95, opacity: 0, duration: 0.6 }, "<");
// ← image starts at same time as CTA
Stagger with GSAP:
// Stagger a batch of elements
gsap.from(".card", {
opacity: 0,
y: 20,
stagger: {
each: 0.06, // 60ms between each
from: "start", // or "center", "end", "random", "edges"
ease: "power2.in" // Easing applied to the stagger distribution itself
},
duration: 0.3,
ease: "power2.out",
scrollTrigger: ".card-grid",
});
| Stagger interval | Effect | |-----------------|--------| | < 30ms | Effectively simultaneous — imperceptible | | 40–60ms | Smooth wave — most common, natural | | 60–80ms | Readable cascade, each element distinct | | 80–120ms | Deliberate, each element gets full attention | | > 120ms | Slow — use only for < 4 items |
Longer stagger = more deliberate = higher hierarchy. Use generous stagger for onboarding, hero, and empty states. Use tight stagger for lists, grids, and content sections.
Standard pattern for a 5-element hero section at feel-premium speed:
t=0ms: Eyebrow / label (if present) — fade + translateY(16px), 350ms
t=150ms: Primary headline — fade + translateY(16px), 450ms
t=280ms: Subheadline — fade + translateY(12px), 380ms
t=380ms: CTA button(s) — fade + scale(0.97→1), 300ms
t=280ms: Hero image (concurrent with subheadline) — fade + scale(0.97→1), 500ms
Total sequence: ~780ms from first element to fully rendered — short enough to not feel slow, long enough to feel crafted.
Exits are the reverse of entrances — but should run in reverse order and at ~70–80% of entrance duration. The last thing to enter is the first to leave.
const exitVariants = {
visible: { opacity: 1, y: 0 },
hidden: {
opacity: 0,
y: -8,
transition: { duration: 0.2, ease: [0.4, 0, 1, 1] } // ease-in for exit
},
};
// Use AnimatePresence with mode="wait" or mode="sync" to control exit/entry overlap
<AnimatePresence mode="wait">
{isVisible && (
<motion.div
key="content"
initial="hidden"
animate="visible"
exit="hidden"
variants={containerVariants}
>
{children}
</motion.div>
)}
</AnimatePresence>
| Mistake | Consequence | Fix |
|---------|-------------|-----|
| Everything at t=0 | Visual explosion, no narrative | Add staggerChildren or explicit delays |
| Stagger > 100ms for 8+ items | Sequence too slow to watch | Cap at 60ms for large sets |
| Different easing per element | Incoherent feel | Use a shared defaults in GSAP or shared variant config |
| Entrance without exit | Spatial confusion on unmount | Always pair AnimatePresence + exit variants |
| Mixing simultaneous and staggered randomly | No clear read order | Define the visual hierarchy first, then assign delays accordingly |
references/choreography-patterns.md — Page transition choreography, tab switch sequences, modal open/close, loading → content reveal transitions, and a full library of named stagger patterns (cascade, wave, from-center, diagonal, random)testing
Performs quality control on single-cell RNA-seq data (.h5ad or .h5 files) using scverse best practices with MAD-based filtering and comprehensive visualizations. Use when users request QC analysis, filtering low-quality cells, assessing data quality, or following scverse/scanpy best practices for single-cell analysis.
tools
Deep learning for single-cell analysis using scvi-tools. This skill should be used when users need (1) data integration and batch correction with scVI/scANVI, (2) ATAC-seq analysis with PeakVI, (3) CITE-seq multi-modal analysis with totalVI, (4) multiome RNA+ATAC analysis with MultiVI, (5) spatial transcriptomics deconvolution with DestVI, (6) label transfer and reference mapping with scANVI/scArches, (7) RNA velocity with veloVI, or (8) any deep learning-based single-cell method. Triggers include mentions of scVI, scANVI, totalVI, PeakVI, MultiVI, DestVI, veloVI, sysVI, scArches, variational autoencoder, VAE, batch correction, data integration, multi-modal, CITE-seq, multiome, reference mapping, latent space.
testing
This skill should be used when scientists need help with research problem selection, project ideation, troubleshooting stuck projects, or strategic scientific decisions. Use this skill when users ask to pitch a new research idea, work through a project problem, evaluate project risks, plan research strategy, navigate decision trees, or get help choosing what scientific problem to work on. Typical requests include "I have an idea for a project", "I'm stuck on my research", "help me evaluate this project", "what should I work on", or "I need strategic advice about my research".
development
Run nf-core bioinformatics pipelines (rnaseq, sarek, atacseq) on sequencing data. Use when analyzing RNA-seq, WGS/WES, or ATAC-seq data—either local FASTQs or public datasets from GEO/SRA. Triggers on nf-core, Nextflow, FASTQ analysis, variant calling, gene expression, differential expression, GEO reanalysis, GSE/GSM/SRR accessions, or samplesheet creation.