import React, { useRef, useEffect, useCallback } from 'react';
import {
starryNight, floatingBubbles,
gradientWave, particleNetwork, galaxySpiral,
rainbowWaves, geometricShapes, fireflies,
matrixRain, quantumField, electricStorm,
cosmicDust, neonPulse, auroraBorealis, oceanWaves,
neuralNetwork,
dnaHelix, snowFall, realisticRain, autumnLeaves, realisticClouds, fireflyForest, fallingFoodFiesta,
hauntedForest,
ghostlyApparitions,
spiderwebOverlay,
undeadGraveyard,
bloodRain,
creepyCrawlies,
} from './backgroundAnimations';
import AnimatedText from './AnimatedText';
/**
* @module AnimatedBackground
* @description A React component that creates animated canvas backgrounds with blend modes
*/
/**
* AnimatedBackground Component
* @param {Object} props - Component props
* @param {string} props.animationName - Name of the animation to display
* @param {string} [props.fallbackAnimation='geometricShapes'] - Fallback animation if main animation fails
* @param {number} [props.fps=60] - Frames per second for the animation
* @param {string} [props.blendMode='normal'] - Canvas blend mode
* @param {Object} [props.style] - Additional CSS styles for the canvas
* @returns {React.Component} A canvas element with the animated background
*/
const AnimatedBackground = ({
animationName,
fallbackAnimation = 'geometricShapes',
fps = 60,
blendMode = 'normal',
style
}) => {
const canvasRef = useRef(null);
const animationRef = useRef(null);
const blendModes = [
'normal',
'multiply',
'screen',
'overlay',
'darken',
'lighten',
'color-dodge',
'color-burn',
'hard-light',
'soft-light',
'difference',
'exclusion',
'hue',
'saturation',
'color',
'luminosity'
];
const setupCanvas = useCallback(() => {
const canvas = canvasRef.current;
const ctx = canvas.getContext('2d');
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;
// Set blend mode
ctx.globalCompositeOperation = blendModes.includes(blendMode)
? blendMode
: 'normal';
const animations = {
starryNight, floatingBubbles, gradientWave, particleNetwork, galaxySpiral,
rainbowWaves, geometricShapes, fireflies, matrixRain, quantumField,
electricStorm, cosmicDust, neonPulse, auroraBorealis, oceanWaves,
neuralNetwork, dnaHelix, snowFall, realisticRain, realisticClouds,
fireflyForest, autumnLeaves, fallingFoodFiesta, hauntedForest,
ghostlyApparitions,
spiderwebOverlay,
undeadGraveyard,
bloodRain,
creepyCrawlies,
};
let animation = animations[animationName];
if (!animation) {
console.warn(`Animation "${animationName}" not found. Using fallback animation.`);
animation = animations[fallbackAnimation] || animations.geometricShapes;
}
return animation(canvas, ctx);
}, [animationName, fallbackAnimation, blendMode]);
useEffect(() => {
const animate = () => {
const animation = setupCanvas();
let lastTime = 0;
const frameInterval = 1000 / fps;
const loop = (currentTime) => {
animationRef.current = requestAnimationFrame(loop);
const deltaTime = currentTime - lastTime;
if (deltaTime >= frameInterval) {
lastTime = currentTime - (deltaTime % frameInterval);
animation();
}
};
animationRef.current = requestAnimationFrame(loop);
};
animate();
const handleResize = () => {
if (canvasRef.current) {
canvasRef.current.width = window.innerWidth;
canvasRef.current.height = window.innerHeight;
setupCanvas();
}
};
window.addEventListener('resize', handleResize);
return () => {
if (animationRef.current) {
cancelAnimationFrame(animationRef.current);
}
window.removeEventListener('resize', handleResize);
};
}, [fps, setupCanvas]);
return (
<canvas
ref={canvasRef}
style={{
position: 'fixed',
top: 0,
left: 0,
width: '100%',
height: '100%',
zIndex: -1,
...style
}}
/>
);
};
export {
AnimatedBackground,
starryNight,
floatingBubbles,
gradientWave,
particleNetwork,
galaxySpiral,
rainbowWaves,
geometricShapes,
fireflies,
matrixRain,
quantumField,
electricStorm,
cosmicDust,
neonPulse,
auroraBorealis,
fallingFoodFiesta,
hauntedForest,
ghostlyApparitions,
spiderwebOverlay,
undeadGraveyard,
bloodRain,
creepyCrawlies,
AnimatedText,
};
Source