Let me show you how I created a CSS-only image glitch effect. I was working on the robot poet and wanted my robotic bard to glitch - because it felt fitting given the quality of poetry it generates.
Here's the final result:
The effect involves quite a bit of HTML and CSS but no JavaScript. I did use JavaScript to generate the HTML and CSS, but it is not used in the final version.
To be clear, there's nothing wrong with using JavaScript for this kind of thing. I just saw it as a fun challenge to make a pure CSS version.
The idea
In the words of Daft Punk:
Slice It
Move It
Hue-Rotate It
We'll have to slice the image into strips and randomly displace them while altering colors at the same time.
Slice it
I wanted to use a single image without having to slice it manually.
To achieve that, we need to create a bunch of divs. Each div represents one strip and has the image set as a background, but the image is shifted vertically. When stacked on top of each other, the divs look the same as the original image. To make the glitch effect more believable, we'll use a random height for each strip.
We could do this by hand, but that would be tedious, so let's use code:
This gives us a list of divs with the image inside. When we render them all, we get the original image in a bunch of HTML elements we can manipulate individually.
Now that we have our strips, let's try moving them around.
Move it
CSS animations, interpolate values smoothly (I'm aware of steps timing function, but it doesn't work in this case as it takes interpolated values at each step.) . And a smooth animation wouldn't feel glitchy - it needs to snap between frames instead. The quick, erratic movement will help it be convincing.
Default easing
Let's start with a simple example of a smooth animation. We'll move a div horizontally for 100 pixels, keep it there for a short time, and then move it back to the starting position.
{
33%,
66% }
}
Try dragging the timeline, all graphs are interactive. (Instead of making a few static images, I got sidetracked, and wrote a whole library that generates these interactive SVG timelines. I named it Monorail, and plan to release it soon.)
This kind of works, but it is not the intended effect. To achieve instant movement, we need to add more keyframes. The closer the keyframes are to each other, the snappier the movement will be. If we put them really close, the movement will be super short - shorter than it takes to render a single frame. This will give us the sudden movement we are after using only CSS.
Let's try doing that.
Super short keyframes
Now we'll add keyframes really close to the ones from the previous example. Notice how it created the almost vertical jump between two states, compared to the smooth line we had before. The line will never be truly vertical, but if the keyframes are really close, the movement will happen in a single frame, achieving our desired effect.
{
0%,
32.9%,
66.1%,
100% }
33%,
66% }
}
Again, try playing with the timeline to see how the animation has changed.
One thing to keep in mind: keyframes are defined in percentage values, so their duration depends on the total animation duration. If we define a keyframe of 0.1% and the animation lasts for 5 seconds, the keyframe is going to last for 5 milliseconds. That is shorter than even a 120fps refresh rate, so I think we are pretty safe. Even if the movement lasts for a few frames, our eyes can't really notice it (at least mine can't).
Now let's mess up some colors.
Hue-rotate it
If you imagine the color wheel, hue-rotation shifts all the colors by the same amount. Applying a random value creates a distorted look. My plan was to combine this effect with movement to make the distortion feel even stronger.
Implementing it was supposed to be the easiest part. Add hue-rotate to the animation and call it a day. But alas, in Safari, property worked on its own, but not when animated. I'll spare you the details of everything I tried, but finally I found that changing the initial state from none
to hue-rotate(0)
instead of none
solved the issue.
This works:
{
0%,
5% }
10% }
}
And this doesn't:
{
0%,
5% }
10% }
}
Once I solved it, I didn't bother to dig deeper. I've seen people get angry about browser issues, but honestly, I don't blame Safari developers. Imagine building and maintaining a browser - it is like an operating system inside of an operating system. A bug like this slipping through the cracks is completely understandable.
Anyway, back to our glitch effect - let's try to combine transform and filter in a single animation.
Stripe animation
Now we need to apply everything we learned to our strips. I decided to have two jumps for each strip, along with the hue-rotate effect. Let's see how a single strip looks like fully animated.
Now to add some randomness, otherwise the effect will be too predictable and boring.
Randomness
CSS doesn't support random values on its own. To introduce randomness into our glitch effect, we'll use JavaScript to generate multiple variations of the CSS keyframes animations.
Here's how it works:
- We use JavaScript to generate several animations, each using different keyframe values, but using the same CSS variables (
--glitch-x-1
,--glitch-x-2
,--glitch-hue-1
and--glitch-hue-2
). - When we generate strips HTML, we randomly assign values to these CSS variables, as well as animation properties (like duration and delay)
This combination of dynamically generated animations, CSS variables and inline styles gives each strip a unique glitch animation, resulting in the randomized effect we were aiming for.
For example, one of the generated CSS animations will look like this:
{
0.00%,
33.33%,
43.33%,
66.67%,
76.67%,
100.00% }
33.43%,
43.23% }
66.77%,
76.57% }
}
And after generating random values and inlining them, this is how HTML for a single strip looks like:
Once HTML and CSS code is generated, we can use it on its own, without any JavaScript.
Chromatic aberration
You might have noticed that in the final version, I sneaked in subtle blue and red shadows to give impression of color separation, like on a corrupted screen. I had no idea what this effect was called, but a friend who proofread this post said:
You can add something about chromatic aberration.
Well, if you want to learn more about it, now you know what to search for.
There you have it
Is this the best glitch effect you've seen? Probably not. But it is mine, and it runs entirely on CSS, no JavaScript at all. Another cool thing is that this effect works on any content, not just images. As long as you prepare your styles correctly, you can use it on any HTML content.
You can see the code and play with the final version on CodePen.
If you end up using this effect or the discrete animation hack, please let me know - I'd love to see it in action.
And before you go, here are other two examples of glitching effects I found to be particularly interesting:
Thank you for reading and happy glitching!