All components
Liquid Gradient
gradientsUi-Layouts component.
responsive · 600px
Install
Same command in any shadcn project — React (Vite/CRA), Next.js, Remix, Astro, and more:
$
npx shadcn@latest add https://your-domain/r/liquid-gradient.jsonUsage
'use client';
import { useState } from 'react';
import { Liquid } from '@/registry/ui-layouts/liquid-gradient';
const sampleColors = {
color1: '#ff6b6b',
color2: '#ffa36b',
color3: '#ffcc6b',
color4: '#a8ff6b',
color5: '#6bffa3',
color6: '#6bccff',
color7: '#6b8cff',
color8: '#a36bff',
color9: '#ff6bcf',
color10: '#ff6b8c',
color11: '#ff9f6b',
color12: '#6bffcc',
color13: '#6bffd4',
color14: '#6be8ff',
color15: '#8c6bff',
color16: '#cf6bff',
color17: '#ff6be8',
};
export default function Demo() {
const [isHovered, setIsHovered] = useState(false);
return (
<div className="w-full max-w-2xl mx-auto flex items-center justify-center p-8">
<div
className="relative w-[400px] h-[120px] overflow-hidden rounded-[60px] cursor-pointer"
onMouseEnter={() => setIsHovered(true)}
onMouseLeave={() => setIsHovered(false)}
>
<Liquid isHovered={isHovered} colors={sampleColors} />
<span className="absolute inset-0 flex items-center justify-center text-white font-semibold text-lg z-10 select-none">
Hover me
</span>
</div>
</div>
);
}Component source
'use client';
import { AnimatePresence, motion } from 'motion/react';
type ColorKey =
| 'color1'
| 'color2'
| 'color3'
| 'color4'
| 'color5'
| 'color6'
| 'color7'
| 'color8'
| 'color9'
| 'color10'
| 'color11'
| 'color12'
| 'color13'
| 'color14'
| 'color15'
| 'color16'
| 'color17';
export type Colors = Record<ColorKey, string>;
const svgOrder = ['svg1', 'svg2', 'svg3', 'svg4', 'svg3', 'svg2', 'svg1'] as const;
type SvgKey = (typeof svgOrder)[number];
type Stop = {
offset: number;
stopColor: string;
};
type SvgState = {
gradientTransform: string;
stops: Stop[];
};
type SvgStates = Record<SvgKey, SvgState>;
const createStopsArray = (
svgStates: SvgStates,
svgOrder: readonly SvgKey[],
maxStops: number
): Stop[][] => {
const stopsArray: Stop[][] = [];
for (let i = 0; i < maxStops; i++) {
const stopConfigurations = svgOrder.map((svgKey) => {
const svg = svgStates[svgKey];
return svg.stops[i] || svg.stops[svg.stops.length - 1];
});
stopsArray.push(stopConfigurations);
}
return stopsArray;
};
type GradientSvgProps = {
className: string;
isHovered: boolean;
colors: Colors;
};
const GradientSvg: React.FC<GradientSvgProps> = ({ className, isHovered, colors }) => {
const svgStates: SvgStates = {
svg1: {
gradientTransform: 'translate(287.5 280) rotate(-29.0546) scale(689.807 1000)',
stops: [
{ offset: 0, stopColor: colors.color1 },
{ offset: 0.188423, stopColor: colors.color2 },
{ offset: 0.260417, stopColor: colors.color3 },
{ offset: 0.328792, stopColor: colors.color4 },
{ offset: 0.328892, stopColor: colors.color5 },
{ offset: 0.328992, stopColor: colors.color1 },
{ offset: 0.442708, stopColor: colors.color6 },
{ offset: 0.537556, stopColor: colors.color7 },
{ offset: 0.631738, stopColor: colors.color1 },
{ offset: 0.725645, stopColor: colors.color8 },
{ offset: 0.817779, stopColor: colors.color9 },
{ offset: 0.84375, stopColor: colors.color10 },
{ offset: 0.90569, stopColor: colors.color1 },
{ offset: 1, stopColor: colors.color11 },
],
},
svg2: {
gradientTransform: 'translate(126.5 418.5) rotate(-64.756) scale(533.444 773.324)',
stops: [
{ offset: 0, stopColor: colors.color1 },
{ offset: 0.104167, stopColor: colors.color12 },
{ offset: 0.182292, stopColor: colors.color13 },
{ offset: 0.28125, stopColor: colors.color1 },
{ offset: 0.328792, stopColor: colors.color4 },
{ offset: 0.328892, stopColor: colors.color5 },
{ offset: 0.453125, stopColor: colors.color6 },
{ offset: 0.515625, stopColor: colors.color7 },
{ offset: 0.631738, stopColor: colors.color1 },
{ offset: 0.692708, stopColor: colors.color8 },
{ offset: 0.75, stopColor: colors.color14 },
{ offset: 0.817708, stopColor: colors.color9 },
{ offset: 0.869792, stopColor: colors.color10 },
{ offset: 1, stopColor: colors.color1 },
],
},
svg3: {
gradientTransform: 'translate(264.5 339.5) rotate(-42.3022) scale(946.451 1372.05)',
stops: [
{ offset: 0, stopColor: colors.color1 },
{ offset: 0.188423, stopColor: colors.color2 },
{ offset: 0.307292, stopColor: colors.color1 },
{ offset: 0.328792, stopColor: colors.color4 },
{ offset: 0.328892, stopColor: colors.color5 },
{ offset: 0.442708, stopColor: colors.color15 },
{ offset: 0.537556, stopColor: colors.color16 },
{ offset: 0.631738, stopColor: colors.color1 },
{ offset: 0.725645, stopColor: colors.color17 },
{ offset: 0.817779, stopColor: colors.color9 },
{ offset: 0.84375, stopColor: colors.color10 },
{ offset: 0.90569, stopColor: colors.color1 },
{ offset: 1, stopColor: colors.color11 },
],
},
svg4: {
gradientTransform: 'translate(860.5 420) rotate(-153.984) scale(957.528 1388.11)',
stops: [
{ offset: 0.109375, stopColor: colors.color11 },
{ offset: 0.171875, stopColor: colors.color2 },
{ offset: 0.260417, stopColor: colors.color13 },
{ offset: 0.328792, stopColor: colors.color4 },
{ offset: 0.328892, stopColor: colors.color5 },
{ offset: 0.328992, stopColor: colors.color1 },
{ offset: 0.442708, stopColor: colors.color6 },
{ offset: 0.515625, stopColor: colors.color7 },
{ offset: 0.631738, stopColor: colors.color1 },
{ offset: 0.692708, stopColor: colors.color8 },
{ offset: 0.817708, stopColor: colors.color9 },
{ offset: 0.869792, stopColor: colors.color10 },
{ offset: 1, stopColor: colors.color11 },
],
},
};
const maxStops = Math.max(...Object.values(svgStates).map((svg) => svg.stops.length));
const stopsAnimationArray = createStopsArray(svgStates, svgOrder, maxStops);
const gradientTransform = svgOrder.map((svgKey) => svgStates[svgKey].gradientTransform);
const variants = {
hovered: {
gradientTransform: gradientTransform,
transition: { duration: 50, repeat: Number.POSITIVE_INFINITY, ease: 'linear' as const },
},
notHovered: {
gradientTransform: gradientTransform,
transition: { duration: 10, repeat: Number.POSITIVE_INFINITY, ease: 'linear' as const },
},
};
return (
<svg
className={className}
width='1030'
height='280'
viewBox='0 0 1030 280'
fill='none'
xmlns='http://www.w3.org/2000/svg'
>
<rect width='1030' height='280' rx='140' fill='url(#paint0_radial_905_231)' />
<defs>
<motion.radialGradient
id='paint0_radial_905_231'
cx='0'
cy='0'
r='1'
gradientUnits='userSpaceOnUse'
animate={isHovered ? variants.hovered : variants.notHovered}
>
{stopsAnimationArray.map((stopConfigs, index) => (
<AnimatePresence key={index}>
<motion.stop
initial={{
offset: stopConfigs[0].offset,
stopColor: stopConfigs[0].stopColor,
}}
animate={{
offset: stopConfigs.map((config) => config.offset),
stopColor: stopConfigs.map((config) => config.stopColor),
}}
transition={{
duration: 0,
ease: 'linear',
repeat: Number.POSITIVE_INFINITY,
}}
/>
</AnimatePresence>
))}
</motion.radialGradient>
</defs>
</svg>
);
};
type LiquidProps = {
isHovered: boolean;
colors: Colors;
buttonType?: boolean;
};
export const Liquid: React.FC<LiquidProps> = ({ isHovered, colors, buttonType }) => {
return (
<>
{Array.from({ length: 7 }).map((_, index) => (
<div
key={index}
className={`absolute ${index < 3 ? 'w-[443px] h-[121px]' : 'w-[756px] h-[207px]'} ${
index === 0
? 'top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 mix-blend-difference'
: index === 1
? 'top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 rotate-[164.971deg] mix-blend-difference'
: index === 2
? 'top-1/2 left-1/2 -translate-x-[53%] -translate-y-[53%] rotate-[-11.61deg] mix-blend-difference'
: index === 3
? 'top-1/2 left-1/2 -translate-x-1/2 -translate-y-[57%] rotate-[-179.012deg] mix-blend-difference'
: index === 4
? 'top-1/2 left-1/2 -translate-x-[57%] -translate-y-1/2 rotate-[-29.722deg] mix-blend-difference'
: index === 5
? 'top-1/2 left-1/2 -translate-x-[62%] -translate-y-[24%] rotate-[160.227deg] mix-blend-difference'
: 'top-1/2 left-1/2 -translate-x-[67%] -translate-y-[29%] rotate-180 mix-blend-hard-light'
}`}
>
<GradientSvg className='w-full h-full' isHovered={isHovered} colors={colors} />
</div>
))}
</>
);
};Dependencies
motion
Source: Ui-Layouts