react-sparkle
Version:
A React component to increase the number of sparkles in your app
354 lines (312 loc) • 11.5 kB
JavaScript
/* globals window Image */
// If we refactor this:
// * use functional components and hooks
// * extract canvas logic and make it more testable
// * resolve disabled eslint rules
// * breaking: possibly make Sparkle the parent component so we can
// remove resize-observer and CSS position requirements
import React from 'react'
import { ResizeObserver } from '@juggle/resize-observer'
const spriteSrc =
''
const spriteCoords = [0, 6, 13, 20]
const flickerSpeedConstants = {
slowest: 50,
slower: 20,
slow: 12,
normal: 7,
fast: 4,
faster: 2,
fastest: 0,
}
// Inspired by and drawn from:
// https://github.com/simeydotme/jQuery-canvas-sparkles
class Sparkle extends React.Component {
// Returns a value from spriteCoords, determining which slice of
// the sprite we display
static getSpriteVariant() {
return spriteCoords[Math.floor(Math.random() * spriteCoords.length)]
}
static getOpacity() {
return Math.random()
}
static randomHexColor() {
// http://www.paulirish.com/2009/random-hex-color-code-snippets/
return `#${`000000${Math.floor(Math.random() * 16777215).toString(
16
)}`.slice(-6)}`
}
constructor(props) {
super(props)
this.sparkleWrapper = null
this.sparkleCanvas = null
this.sparkleContext = null
this.sparkles = []
this.animationFrame = null
this.sprite = null
}
componentDidMount() {
this.init()
}
componentWillUnmount() {
this.end()
}
getColor() {
const { color } = this.props
let chosenColor
if (color === 'random') {
chosenColor = Sparkle.randomHexColor()
// Check if is an array
} else if (
(Array.isArray && Array.isArray(color)) ||
color instanceof Array
) {
// Choose a random color from the array
chosenColor = color[Math.floor(Math.random() * color.length)]
} else {
chosenColor = color
}
return chosenColor
}
randomSparkleSize() {
const { minSize, maxSize } = this.props
return Math.floor(Math.random() * (maxSize - minSize + 1) + minSize)
}
// Assigns fresh values to an existing sparkle
recreateSparkle(existingSparkle) {
if (!this.sparkleCanvas) {
return null
}
const size = this.randomSparkleSize()
return Object.assign(existingSparkle, {
// Subtract size so sparkles don't get cut off by the edge of the canvas
position: {
x: Math.floor(Math.random() * (this.sparkleCanvas.width - size)),
y: Math.floor(Math.random() * (this.sparkleCanvas.height - size)),
},
size,
opacity: Sparkle.getOpacity(),
color: this.getColor(),
variant: Sparkle.getSpriteVariant(),
})
}
createSparkle() {
return this.recreateSparkle({})
}
createSparkles() {
const { count } = this.props
// Create `this.props.count` number of sparkles
for (let i = 0; i < count; i += 1) {
this.sparkles.push(this.createSparkle())
}
}
drawSparkles() {
if (!this.sparkleCanvas || !this.sparkleContext) {
return
}
// Clear canvas
this.sparkleContext.clearRect(
0,
0,
this.sparkleCanvas.width,
this.sparkleCanvas.height
)
const self = this
// Draw each sparkle
this.sparkles.forEach((sparkle) => {
self.sparkleContext.save()
self.sparkleContext.globalAlpha = sparkle.opacity
self.sparkleContext.drawImage(
this.sprite,
sparkle.variant, // show different sparkle styles
0,
7,
7,
sparkle.position.x,
sparkle.position.y,
sparkle.size,
sparkle.size
)
// Tint with the color
if (sparkle.color) {
self.sparkleContext.globalCompositeOperation = 'source-atop'
self.sparkleContext.globalAlpha = 0.6
self.sparkleContext.fillStyle = sparkle.color
self.sparkleContext.fillRect(
sparkle.position.x,
sparkle.position.y,
sparkle.size,
sparkle.size
)
}
self.sparkleContext.restore()
})
}
updateSparkles() {
const { flicker, flickerSpeed, fadeOutSpeed, newSparkleOnFadeOut } =
this.props
const self = this
this.animationFrame = window.requestAnimationFrame((time) => {
// Integer of current time. Useful for events that we want to do
// less frequently than any animation frame.
const currentTimeInt = Math.floor(time)
// Update sparkles by doing some or all of the following:
// - change opacity
// - change position
// - change sprite slice to add "flicker" effect
this.sparkles.forEach((sparkle) => {
// If we refactor this, don't reassign to the sparkle param.
// eslint-disable-next-line no-param-reassign
sparkle.opacity -= 0.001 * fadeOutSpeed
// Sometimes change the sparkle variant for a "flicker" effect
if (flicker) {
const flickerSpeedConstant = flickerSpeedConstants[flickerSpeed]
if (
currentTimeInt %
Math.floor(Math.random() * flickerSpeedConstant + 1) ===
0
) {
// eslint-disable-next-line no-param-reassign
sparkle.variant = Sparkle.getSpriteVariant()
}
}
// Sparkle has faded out
if (sparkle.opacity < 0) {
// Either replace the sparkle with a brand new one or
// reset the opacity
if (newSparkleOnFadeOut) {
self.recreateSparkle(sparkle)
} else {
// eslint-disable-next-line no-param-reassign
sparkle.opacity = Sparkle.getOpacity()
}
}
})
// Draw the updated sparkles
self.drawSparkles()
// Continue to update sparkles
self.updateSparkles()
})
}
// Resize our canvas when the parent resizes
parentResizeObserver() {
const parentStyle = window.getComputedStyle(this.sparkleWrapper.parentNode)
const boxSizing = parentStyle['box-sizing']
const isHorizontalWritingMode =
parentStyle['writing-mode'] === 'horizontal-tb'
const self = this
const ro = new ResizeObserver((entries) => {
// If we refactor this, use an array iteration instead.
// eslint-disable-next-line no-restricted-syntax
for (const entry of entries) {
const { blockSize, inlineSize } = entry.borderBoxSize[0]
const [width, height] = isHorizontalWritingMode
? [inlineSize, blockSize]
: [blockSize, inlineSize]
self.sizeCanvas(width, height)
}
})
ro.observe(this.sparkleWrapper.parentNode, { box: boxSizing })
}
// Used in ResizeObserver
// eslint-disable-next-line react/no-unused-class-component-methods
sizeCanvas(parentWidth, parentHeight) {
if (!this.sparkleCanvas) {
return
}
const { overflowPx } = this.props
// Size the canvas
this.sparkleCanvas.width = parentWidth + 2 * overflowPx
this.sparkleCanvas.height = parentHeight + 2 * overflowPx
}
start() {
this.createSparkles()
this.drawSparkles()
this.updateSparkles()
}
end() {
window.cancelAnimationFrame(this.animationFrame)
this.sparkles = []
}
init() {
if (!this.sparkleCanvas) {
// eslint-disable-next-line no-console
console.warn('No sparkles today :( The canvas did not render.')
return
}
// Create sprite
const sprite = new Image()
sprite.src = spriteSrc
this.sprite = sprite
this.sparkleContext = this.sparkleCanvas.getContext('2d')
this.parentResizeObserver()
this.start()
}
render() {
const { overflowPx } = this.props
return (
<span
ref={(sparkleWrapper) => {
this.sparkleWrapper = sparkleWrapper
}}
style={{
width: '100%',
height: '100%',
overflow: 'visible',
position: 'absolute',
top: `-${overflowPx}px`,
left: `-${overflowPx}px`,
pointerEvents: 'none',
}}
>
<canvas
ref={(canvas) => {
this.sparkleCanvas = canvas
}}
/>
</span>
)
}
}
Sparkle.defaultProps = {
// The color of the sparkles. Can be a color, an array of colors,
// or 'random' (which will randomly pick from all hex colors).
color: '#FFF',
// The number of sparkles to render. A large number could slow
// down the page.
count: 50,
// The minimum and maximum diameter of sparkles, in pixels.
minSize: 5,
maxSize: 8,
// The number of pixels the sparkles should extend beyond the
// bounds of the parent element.
overflowPx: 20,
// How quickly sparkles disappear; in other words, how quickly
// new sparkles are created. Should be between 0 and 1000,
// with 0 never fading sparkles out and 1000 immediately
// removing sparkles. Most meaningful speeds are between
// 0 and 150.
fadeOutSpeed: 50,
// Whether we should create an entirely new sparkle when one
// fades out. If false, we'll just reset the opacity, keeping
// all other attributes of the sparkle the same.
newSparkleOnFadeOut: true,
// Whether sparkles should have a "flickering" effect.
flicker: true,
// How quickly the "flickering" should happen.
// One of: slowest, slower, slow, normal, fast, faster, fastest
flickerSpeed: 'normal',
}
// Features that would be good to add:
// - "active" prop to turn on and off
// - Option to fade in new sparkles
// - Sparkle movement
// - "Wandering" movement, as in https://github.com/simeydotme/jQuery-canvas-sparkles
// - Function-based movement likelihood (e.g. gravity-esque behavior)
// - Recreate sparkles when they leave the canvas
// - Larger-sized sparkles that still look good (the existing
// sprites get blurry); possibly use drawn canvas images
// instead of sprites
// - Declarative updates for the number of sparkles, canvas size,
// and fade out speed
export default Sparkle