Spotlight effects can be a pretty creative way of revealing content on your website. And they're surprisingly easy to create with a little JS and CSS!
The basic idea
TLDR; Use CSS radial gradient backgrounds, and then use JS to track the mouse movement and move the center of the gradient.
There are other, more complex ways of doing this - you could use canvas or external libraries - but I find the simpler approach is a good starting point and should work well enough for most use cases.
Let's get started:
#spotlight { position: fixed; opacity: 1; width: 100vw; height: 100vh; top: 0; left: 0; pointer-events: none; }
<div id="spotlight"></div>
const spotlightEl = document.querySelector("#spotlight"); function handleMouseMove(event) { const { clientX, clientY } = event; spotlightEl.style.background = `radial-gradient(circle at ${clientX}px ${clientY}px, #00000000 10px, #000000ee 350px)`; } document.addEventListener("mousemove", handleMouseMove)
And...that's pretty much it.
The #spotlight
div is positioned so that it always covers the entire viewport. We then attach an event listener to listen to the mouse's movement, and set the X
and Y
directly in the radial-gradient
CSS value.
The #00000000 10px, #000000ee 350px
part basically means we want a gradient that is transparent at it's center, and at 350px from the center is a slightly transparent black. That's how the gradient center reveals what's beneath.
Making it configurable
So far we have a basic spotlight, so let's go ahead and improve it.
We can create a Spotlight
class that allows you to easily set up new spotlights and pass in some options:
toggleEl
is the id of the element you want to use as a trigger to toggle the spotlight.innerRadius
andouterRadius
set the spotlight size.outerColor
sets the background color (try passing inred
for a truly bleeding edge experience).
The class also provides methods for switching the spolight on and off.
Making the light feel more natural
To make it a look a little more natural, there are a couple of added touches.
First, there is a slight delay in the spotlight movement. This adds a sense of weight to the light, like we're dragging it around with our mouse.
handleMouseMove(event: MouseEvent) { setTimeout(() => { this.updateEl(event.clientX, event.clientY); }, 50); }
Also, you'll notice the light "pulses" slightly - continually increasing and decreasing.
But how do we get it to work?
My first instinct was to animate the gradient size, but that's not actually possible in CSS! Next I thought of somehow changing the outerRadius
value in steps inside a function called in a setInterval
, but that turned out to be a really stupid idea. Finally, I came up with the simple solution - animate the scale of the entire spotlight div with a CSS animation.
@keyframes pulse { 0% { transform: scale(1); } 100% { transform: scale(1.1); } }
this.el.style.animation = "pulse 3s ease-in-out infinite alternate forwards";
This way, we don't need to touch the actual gradient, we just animate the entire "canvas", stretching the gradient in the process. I also used CSS animations for the "switching on/off" part.
Wrapping up
Obviously, you shouldn't be using spotlights in every single page (please don't!). But when used right, they're a nice way for users to feel like they're discovering some cool concealed content for the first time.
Thanks for reading!