Files
claudemesh/apps/web/src/modules/tts/speech/avatar.tsx
Alejandro Gutiérrez d3163a5bff feat(db): mesh data model — meshes, members, invites, audit log
- pgSchema "mesh" with 4 tables isolating the peer mesh domain
- Enums: visibility, transport, tier, role
- audit_log is metadata-only (E2E encryption enforced at broker/client)
- Cascade on mesh delete, soft-delete via archivedAt/revokedAt

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 21:19:32 +01:00

199 lines
5.6 KiB
TypeScript

import { motion } from "motion/react";
import {
Avatar,
AvatarFallback,
AvatarImage,
} from "@turbostarter/ui-web/avatar";
import type {Variants} from "motion/react";
import type { UIVoice } from "~/modules/tts/utils/types";
const avatarAnimationVariants: Variants = {
playing: {
scale: [1, 1.04, 0.97, 1.03, 0.98, 1.02, 1],
rotate: [0, 1.8, -1.2, 2.5, -1.5, 0.8, 0],
x: [0, 3, -2, 2, -3, 1, 0],
y: [0, -3, 2, -4, 3, -1, 0],
filter: [
"brightness(1) contrast(1)",
"brightness(1.08) contrast(1.04)",
"brightness(0.98) contrast(0.99)",
"brightness(1.06) contrast(1.03)",
"brightness(0.97) contrast(0.98)",
"brightness(1.04) contrast(1.02)",
"brightness(1) contrast(1)",
],
transition: {
duration: 5,
repeat: Infinity,
ease: "easeInOut" as const,
filter: {
duration: 5,
repeat: Infinity,
ease: "easeInOut" as const,
},
},
},
loading: {
scale: [1, 1.02, 0.99, 1.01, 1],
filter: [
"brightness(0.9) contrast(0.95) grayscale(1)",
"brightness(0.95) contrast(0.97) grayscale(1)",
"brightness(0.88) contrast(0.94) grayscale(1)",
"brightness(0.93) contrast(0.96) grayscale(1)",
"brightness(0.9) contrast(0.95) grayscale(1)",
],
transition: {
duration: 2,
repeat: Infinity,
ease: "easeInOut" as const,
filter: {
duration: 2,
repeat: Infinity,
ease: "easeInOut" as const,
},
},
},
idle: {
scale: 1,
rotate: 0,
x: 0,
y: 0,
filter: "brightness(1) contrast(1) grayscale(0)",
transition: {
duration: 1,
ease: "easeOut" as const,
},
},
};
const imageAnimationVariants: Variants = {
playing: {
scale: [1, 1.03, 0.98, 1.02, 1],
rotate: [0, -0.5, 0.3, -0.2, 0],
filter: [
"saturate(1.1) brightness(1) grayscale(0)",
"saturate(1.25) brightness(1.05) grayscale(0)",
"saturate(1.15) brightness(0.98) grayscale(0)",
"saturate(1.2) brightness(1.03) grayscale(0)",
"saturate(1.1) brightness(1) grayscale(0)",
],
transition: {
duration: 4,
repeat: Infinity,
ease: "easeInOut" as const,
filter: {
duration: 4,
repeat: Infinity,
ease: "easeInOut" as const,
},
},
},
loading: {
scale: [1, 1.01, 0.99, 1],
filter: [
"saturate(0.8) brightness(0.9) grayscale(1)",
"saturate(0.85) brightness(0.95) grayscale(1)",
"saturate(0.8) brightness(0.88) grayscale(1)",
"saturate(0.8) brightness(0.9) grayscale(1)",
],
transition: {
duration: 1.5,
repeat: Infinity,
ease: "easeInOut" as const,
filter: {
duration: 1.5,
repeat: Infinity,
ease: "easeInOut" as const,
},
},
},
idle: {
scale: 1,
rotate: 0,
filter: "saturate(1.1) grayscale(0)",
transition: {
duration: 1,
ease: "easeOut" as const,
},
},
};
interface VoiceAvatarProps {
readonly voice: UIVoice;
readonly playing: boolean;
readonly loading: boolean;
}
export function VoiceAvatar({ voice, playing, loading }: VoiceAvatarProps) {
const animationState = playing ? "playing" : loading ? "loading" : "idle";
return (
<div className="relative flex grow items-center justify-center">
<motion.div
animate={animationState}
initial="idle"
variants={avatarAnimationVariants}
className="relative z-10 flex h-full max-h-[min(50vw,18rem)] items-center justify-center"
>
<Avatar className="relative aspect-square h-full w-auto">
<AvatarFallback>{voice.name.charAt(0)}</AvatarFallback>
<motion.div
animate={animationState}
initial="idle"
variants={imageAnimationVariants}
style={{ width: "100%", height: "100%" }}
>
<AvatarImage src={voice.avatar?.src} style={voice.avatar?.style} />
</motion.div>
</Avatar>
{!loading && (
<>
<div className="absolute -inset-20 overflow-hidden rounded-full blur-3xl">
<motion.div
className="h-full w-full"
style={{
...voice.avatar?.style,
background: `radial-gradient(circle, hsla(210,100%,55%,0) 0%, hsla(210,100%,55%,0.4) 40%, hsla(210,100%,55%,0.6) 70%, hsla(210,100%,55%,0.2) 100%)`,
mixBlendMode: "soft-light",
}}
animate={{
opacity: [0.5, 0.8, 0.6, 0.75, 0.5],
scale: [0.75, 1.15, 0.95, 1.1, 0.75],
}}
transition={{
duration: 6,
repeat: Infinity,
ease: "easeInOut",
}}
/>
</div>
<div className="absolute -inset-12 overflow-hidden rounded-full blur-xl">
<motion.div
className="h-full w-full"
style={{
...voice.avatar?.style,
background: `radial-gradient(circle, transparent 10%, hsla(210,100%,55%)/0.5) 60%, hsla(210,100%,55%)/0.7) 80%, transparent 100%)`,
}}
animate={{
opacity: [0.6, 0.9, 0.7, 0.85, 0.6],
scale: [0.75, 1.05, 0.95, 1.02, 0.75],
}}
transition={{
duration: 4.5,
repeat: Infinity,
ease: "easeInOut",
delay: 0.5,
}}
/>
</div>
</>
)}
</motion.div>
</div>
);
}