我需要帮助来应用正在褪色的水效果,如下图所示。在图像中,白色颗粒逐渐褪色为蓝色模糊效果。这就是我正在努力实现的目标。
我相信通过快速模糊或单纯形噪声,我们应该能够实现这一目标,但不确定如何继续。
(点击放大)
在 HTML5 画布中实现了带有碎片的瀑布。
const canvas = document.getElementById('fountainCanvas');
const ctx = canvas.getContext('2d');
// Set canvas size
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;
const particles = [];
const splinters = [];
const gravity = 0.1; // Gravity constant
const fountainX = canvas.width / 2;
const fountainY = canvas.height / 2;
// Mouse position
let mouseX = fountainX;
let mouseY = fountainY;
// Particle class
class Particle {
constructor(x, y, angle, isSplinter = false) {
this.x = x;
this.y = y;
const speed = Math.random() * 3 + 2; // Random speed
const spread = Math.random() * 0.4 - 0.2; // Randomize direction slightly
this.vx = isSplinter ?
(Math.random() * 2 - 1) * 3 :
Math.cos(angle + spread) * speed;
this.vy = isSplinter ?
Math.random() * -3 :
Math.sin(angle + spread) * speed;
this.alpha = isSplinter ? 1 : 1; // Opacity
this.radius = isSplinter ? Math.random() : Math.random() + 1; // Size
this.isSplinter = isSplinter;
}
update() {
this.x += this.vx;
this.y += this.vy;
this.vy += gravity; // Apply gravity
this.alpha -= this.isSplinter ? 0.02 : 0.005; // Fade out
// Check if main particles reach the bottom of the canvas
if (!this.isSplinter && this.y >= canvas.height) {
this.createSplinters(); // Create splinters on impact
this.alpha = 0; // Make particle invisible
}
}
createSplinters() {
for (let i = 0; i < 10; i++) {
splinters.push(new Particle(this.x, canvas.height, 0, true));
}
}
draw() {
ctx.beginPath();
ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2);
ctx.fillStyle = `hsla(199, 100%, 67%, ${this.alpha})`; // Blue shade with opacity
ctx.fill();
}
isAlive() {
return this.alpha > 0; // Check if particle is still visible
}
}
// Create particles over time
function emitParticles() {
const angle = Math.atan2(mouseY - fountainY, mouseX - fountainX);
for (let i = 0; i < 5; i++) { // Emit a few particles per frame
particles.push(new Particle(fountainX, fountainY, angle));
}
}
// Animation loop
function animate() {
ctx.fillStyle = 'rgba(0, 0, 0, 0.2)'; // Trail effect
ctx.fillRect(0, 0, canvas.width, canvas.height);
emitParticles(); // Emit new particles continuously
// Update and draw particles
particles.forEach((particle, index) => {
particle.update();
if (!particle.isAlive()) {
particles.splice(index, 1); // Remove dead particles
} else {
particle.draw();
}
});
// Update and draw splinters
splinters.forEach((splinter, index) => {
splinter.update();
if (!splinter.isAlive()) {
splinters.splice(index, 1); // Remove dead splinters
} else {
splinter.draw();
}
});
requestAnimationFrame(animate);
}
// Update mouse position on move
canvas.addEventListener('mousemove', (event) => {
mouseX = event.clientX;
mouseY = event.clientY;
});
// Initialize animation
animate();
canvas {
display: block;
margin: 0 auto;
background: #000;
/* Black background */
}
<canvas id="fountainCanvas"></canvas>
在您的代码中,您使用 alpha 淡化粒子:
ctx.fillStyle = `hsla(199, 100%, 67%, ${this.alpha})`;
但是你的背景是黑色的,所以淡入淡出是从蓝色到黑色
如果你想创造从一种颜色渐变到另一种颜色的错觉,你可以在完全相同的位置绘制两个
arc
,但第一个使用不同的颜色,而它会在背景上,并且应该创建您正在寻找的经验...
const canvas = document.getElementById('fountainCanvas');
const ctx = canvas.getContext('2d');
var particles = []
class Particle {
constructor(x, y) {
this.x = x;
this.y = y;
this.alpha = 1;
this.radius = Math.random() + 15;
}
draw() {
this.alpha -= 0.005;
ctx.beginPath();
ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2);
ctx.fillStyle = `red`;
ctx.fill();
ctx.beginPath();
ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2);
ctx.fillStyle = `hsla(199, 100%, 67%, ${this.alpha})`;
ctx.fill();
}
}
function emitParticles() {
particles= []
for (let i = 2; i < 7; i++) {
particles.push(new Particle(35*i, 20+10*i));
}
}
function animate() {
ctx.reset();
particles.forEach(p => p.draw());
requestAnimationFrame(animate);
}
setInterval(emitParticles, 4000)
emitParticles()
animate();
canvas {
display: block;
margin: 0 auto;
background: #000;
/* Black background */
}
<canvas id="fountainCanvas"></canvas>
您还可以过渡到其他颜色:
https://www.w3schools.com/colors/colors_hsl.asp
const canvas = document.getElementById('fountainCanvas');
const ctx = canvas.getContext('2d');
var particles = []
class Particle {
constructor(x, y) {
this.x = x;
this.y = y;
this.alpha = 1;
this.radius = Math.random() + 15;
this.color = 199
}
draw() {
this.alpha -= 0.005;
this.color -= 1;
ctx.beginPath();
ctx.arc(this.x, this.y, this.radius, 0, Math.PI * 2);
ctx.fillStyle = `hsla(${this.color}, 100%, 67%, ${this.alpha})`;
ctx.fill();
}
}
function emitParticles() {
particles= []
for (let i = 2; i < 7; i++) {
particles.push(new Particle(35*i, 20+10*i));
}
}
function animate() {
ctx.reset();
particles.forEach(p => p.draw());
requestAnimationFrame(animate);
}
setInterval(emitParticles, 4000)
emitParticles()
animate();
canvas {
display: block;
margin: 0 auto;
background: #000;
/* Black background */
}
<canvas id="fountainCanvas"></canvas>
// script.js
const container = document.getElementById("waterfall-container");
function createFallingElement() {
// Create a new div element
const waterfallItem = document.createElement("div");
waterfallItem.classList.add("waterfall-item");
// Set a random horizontal position
const randomX = Math.random() * window.innerWidth;
waterfallItem.style.left = `${randomX}px`;
// Append the element to the container
container.appendChild(waterfallItem);
// Remove the element after animation ends to avoid clutter
waterfallItem.addEventListener("animationend", () => {
container.removeChild(waterfallItem);
});
}
// Create new falling elements every 200ms
setInterval(createFallingElement, 200);