Skip to content

Instantly share code, notes, and snippets.

@kristojorg
Created March 3, 2017 16:29
Show Gist options
  • Save kristojorg/b4a367910cce346b15593bda77625a2d to your computer and use it in GitHub Desktop.
Save kristojorg/b4a367910cce346b15593bda77625a2d to your computer and use it in GitHub Desktop.
A component to add sparkles to a webpage.
import React from 'react'
import getContext from 'recompose/getContext'
const defaultDensity = 0.00008872; // multiply by height and width of canvas.
const defaultColor = ["#c5392a","#C0C0C0"];
const LIFE_SPAN = 1500;
const SPEED_Y = 2;
const SPEED_X = 1;
const SPEED_MULTIPLIER = 1/10;
class sparkle {
constructor ({x,y,color,style,vx,vy, size, toBeDeleted=false}) {
this.x = x;
this.y = y;
this.color = color;
this.style = style;
this.vx = vx;
this.vy = vy;
this.opacity = Math.random();
this.fading = false;
this.size = size;
// this.toBeDeleted = toBeDeleted
}
update = () => {
this.x = this.x + this.vx;
this.y = this.y + this.vy;
// update the opacity such that the sparkle lives 3s
if (!this.fading) {
this.opacity = this.opacity + (1/LIFE_SPAN)*60;
if (this.opacity > 1) this.opacity = 1;
} else {
this.opacity = this.opacity - (1/LIFE_SPAN)*60;
if (this.opacity < 0) this.opacity = 0;
}
if (this.opacity >= 1) this.fading = true;
}
}
const dataUri = "";
class SparkleView extends React.Component {
constructor (props) {
super(props);
this._sparkles = [];
this._canvas = null;
this.sprite = new Image();
this.sprites = [0,6,13,20];
this.sprite.src = dataUri;
this._initCanvas(props.bodyDimensions);
}
_initCanvas = (bodyDimensions) => {
this._canvas = document.createElement('canvas');
this._canvas.style.position = 'absolute';
this._canvas.style.top = '0';
this._canvas.style.pointerEvents = 'none';
this._canvasContext = this._canvas.getContext('2d');
this._resizeCanvas(bodyDimensions);
document.body.appendChild(this._canvas);
window.requestAnimationFrame(this._go);
}
_resizeCanvas = (bodyDimensions) => {
this._canvas.width = window.innerWidth;
this._canvas.height = bodyDimensions.height;
this.defaultCount = this._canvas.width * this._canvas.height * defaultDensity;
this._adjustSparkleDensity();
}
_spawnSparkles = (num = this.defaultCount) => {
for( let i = 0; i < num; i++ ) {
this._sparkles.push(this._newSparkle());
}
}
_adjustSparkleDensity = () => {
const desiredCount = this._canvas.height * this._canvas.width *defaultDensity;
const adjustment = desiredCount - this._sparkles.length;
if (adjustment < 1) {
this._sparkles.splice(0, -1*adjustment);
}
if (adjustment > 1) {
this._spawnSparkles(adjustment);
}
}
_newSparkle = (x, y) => {
const color = defaultColor[ Math.floor(Math.random()*defaultColor.length) ];
const rand0 = Math.random() < 0.5 ? -1 : 1;
const rand1 = Math.random() < 0.5 ? -1 : 1;
return new sparkle({
x: x ? x : Math.floor(Math.random()*this._canvas.width),
y: y ? y : Math.floor(Math.random()*this._canvas.height),
color,
style: this.sprites[ Math.floor(Math.random()*4) ],
size: parseFloat((Math.random()*2).toFixed(2)),
vx: Math.floor(Math.random() * SPEED_X * rand0)* SPEED_MULTIPLIER,
vy: Math.floor(Math.random() * SPEED_Y * rand1)* SPEED_MULTIPLIER,
});
}
_updateSparkles = () => {
let i = this._sparkles.length;
while (i--) {
// for (let i = 0; i < this._sparkles.length; i++) {
const sparkle = this._sparkles[i];
sparkle.update();
// might need to check if it is off the canvas or has opacity 0 or something and create some new ones
// check if sparkle has no opacity
if (sparkle.opacity <= 0) {
if (!sparkle.toBeDeleted ) {
this._sparkles[i] = this._newSparkle();
} else {
this._sparkles.splice(i,1);
}
}
if (
sparkle.x > this._canvas.width ||
sparkle.x < 0 ||
sparkle.y > this._canvas.height ||
sparkle.y < 0
) {
if (!sparkle.isMouse) {
this._sparkles[i] = this._newSparkle();
} else {
this._sparkles.splice(i,1);
}
}
}
}
// paint gets called with a time from requestAnimationFrame
_paint = (time) => {
// get the context
const ctx = this._canvasContext;
// clear the area
ctx.clearRect( 0, 0, this._canvas.width, this._canvas.height );
// loop through the sparkles and draw them
for (const sparkle of this._sparkles) {
// update the type of image if need be
const modulus = Math.floor(Math.random()*7);
if( Math.floor(time) % modulus === 0 ) {
sparkle.style = this.sprites[ Math.floor(Math.random()*4) ];
}
ctx.save();
ctx.globalAlpha = sparkle.opacity;
ctx.drawImage(this.sprite, sparkle.style, 0, 7, 7, sparkle.x, sparkle.y, 7, 7);
ctx.globalCompositeOperation = "source-atop";
ctx.globalAlpha = 0.5;
ctx.fillStyle = sparkle.color;
ctx.fillRect(sparkle.x, sparkle.y, 7, 7);
ctx.restore();
}
}
_go = (time) => {
this._paint(time);
this._updateSparkles();
window.requestAnimationFrame(this._go);
}
componentDidUpdate(prevProps) {
if (prevProps.bodyDimensions.height !== this.props.bodyDimensions.height) {
this._resizeCanvas(this.props.bodyDimensions);
}
}
render () {
return (<div>{this.props.children}</div>);
}
componentWillUnmount () {
document.body.removeChild(this._canvas);
window.removeEventListener('mousemove', this._handleMouseMove);
}
}
export default getContext({bodyDimensions: React.PropTypes.object})(SparkleView);
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment