Rendering Animated .ani Cursors in the Browser
TL:DR: If you'd like to render .ani
files in the browser, I've published ani-cursor
on NPM which makes it possible.
Windows Animated Cursor files (.ani
) are, as the name implies, animated cursor files used by Microsoft Windows. I recently had a reason to try to get them to render on the web and it was a fun experiment so I thought I'd share how I did it.
But first, here's what the result looks like:
.ani cursors, from Super_Mario_Amp_2.wsz by LuigiHann, animating in the browser
Why?
My side project Webamp is an attempt to render classic Winamp skins in the browser. One aspect of the UI that skins could customize, was the cursors when users hovered over elements. Skin authors did this by supplying a collection of cursor files in their skin. Winamp supported both .cur
and .ani
files for this purpose.
Lucky for me, browsers already support the .cur
file type, so supporting those in Webamp was as simple as getting the cursor file as a data URI and then injecting CSS like this into the DOM:
.someElement {
cursor: url([...], auto);
}
However, up until recently, Webamp didn't support .ani
files for two reasons:
Modern browsers don't support
.ani
files nativelyBrowsers don't support animated image formats (gif, apng) as cursors
Source: MDN
Browser don't support .ani
To get around this, we use JavaScript to parse the .ani
file and extract the frames and metadata indicating the order and timing in which to render the frames. .ani
files use the RIFF container format, and someone on NPM has already written a small library riff-file
that can parse RIFF in the browser. riff-file
breaks the file into sections for us:
Block of 9 metadata numbers
A "rate" array showing for how long each frame is rendered (optional)
A "seq" array, showing in which order the frames should be rendered (optional)
An array of frames containing the raw image data for each frame
riff-file
only gives us the byte range of each of these sections within the .ani
file. We then use the byte-data
library to parse the first three sections into JavaScript numbers.
We now have enough information to construct the animation ourselves!
Browsers don't support animated cursors
To get around this, we use a technique I discovered in a four year old forum post. We define a CSS animation where the cursor
property changes over time.
The result looks something like this:
@keyframes ani-cursor-27 {
0% {
cursor: url([...]), auto;
}
9.090909090909092% {
cursor: url([...]), auto;
}
18.181818181818183% {
cursor: url([...]), auto;
}
27.27272727272727% {
cursor: url([...]), auto;
}
36.36363636363637% {
cursor: url([...]), auto;
}
45.45454545454545% {
cursor: url([...]), auto;
}
54.54545454545454% {
cursor: url([...]), auto;
}
63.63636363636363% {
cursor: url([...]), auto;
}
72.72727272727273% {
cursor: url([...]), auto;
}
81.81818181818183% {
cursor: url([...]), auto;
}
90.9090909090909% {
cursor: url([...]), auto;
}
}
#node-with-cursor:hover {
animation: ani-cursor-27 1760ms step-end infinite;
}
Another option I prototyped was to use the Web Animation API to create the animation. This is a lot cleaner since it does not require constructing a CSS string at runtime. However, the CSS approach is a bit nicer for Webamp since it doesn't require us to track the actual DOM nodes of each element that has an animated cursor.
Note: Safari only recently (Nov. 14th, 2020) merged support for animating the cursor
property so this won't work in Safari until their next release. Update: That fix did not work for url()
cursor values. I've filed a followup issue.
Implementation
With solutions to the above challenges in hand, we just need to implement it!
Firstly, we need a way to serialize the individual frame to a data URI. This requires base64 encoding the UInt8Array
we get from riff-file
and supplying the correct mime type:
function cursorUrlFromByteArray(dataArray: Uint8Array) {
const base64 = window.btoa(String.fromCharCode(...dataArray));
return `data:image/x-win-bitmap;base64,${base64}`;
}
Secondly, we have to take the data we parse from the .ani
file and construct a list of keyframes and their associated percentages. If the animation includes a seq
section, some frames may appear more than once in the animation.
const JIFFIES_PER_MS = 1000 / 60;
const sum = nums => reduce((total, value) => total + value, 0);
export function readAni(contents: Uint8Array): AniCursorImage {
const ani = parseAni(contents);
const rate = ani.rate ?? ani.images.map(() => ani.metadata.iDispRate);
const duration = sum(rate);
const frames = ani.images.map((image) => ({
url: curUrlFromByteArray(image),
percents: [] as number[],
}));
let elapsed = 0;
rate.forEach((r, i) => {
const frameIdx = ani.seq ? ani.seq[i] : i;
frames[frameIdx].percents.push((elapsed / duration) * 100);
elapsed += r;
});
return { duration: duration * JIFFIES_PER_MS, frames };
}
Finally, we use this data to construct our CSS string:
let i = 0;
const uniqueId = () => i++;
export function aniCss(selector: string, ani: AniCursorImage): string {
const animationName = `webamp-ani-cursor-${uniqueId()}`;
const keyframes = ani.frames.map(({ url, percents }) => {
const percent = percents.map((num) => `${num}%`).join(", ");
return `${percent} { cursor: url(${url}), auto; }`;
});
return `
@keyframes ${animationName} {
${keyframes.join("\n")}
}
${selector}:hover {
animation: ${animationName} ${ani.duration}ms step-end infinite;
}
`;
}
There are two small details in the CSS that we generate which are worth calling out.
Firstly, we use a timing-function
of step-end
. This is because discrete properties, like cursor, do not actually change at the time specified by the keyframe (10%), but rather when the animation progress has reached the midpoint between keyframes (source). Luckily the animation progress is computed using the timing-function
so we can use step-end
to ensure the cursor image updates immediately when each keyframe percentage is reached.
Secondly, we add a :hover
pseudo selector so that the animation loop only runs when the cursor is visible. This helps us match Winamp's behavior where the animation always restarts when you hover over an element. It may also save some CPU cycles.
And that's about it! The full implementation can be found in the ani-cursor
NPM module which lives in the Webamp repository on GitHub.
Try it out
If you want to see some animated cursors rendering in your browser, check out these skins that feature animated cursors and try hovering your mouse around.
Future Possibilities
It would be cool to see someone use this library to build a website that makes a large collection of .ani
files available online without needing to convert them to another format server side.
Someone could create (or find) a collection of .ani
files on the Internet Archive and then build a simple web app which pulls files directly from the Internet Archive's servers and renders them in an interesting way.
This is the approach I propose in my previous blog post Mainlining Nostalgia: Making the Winamp Skin Museum.
If you end up taking inspiration from this post, please get in touch!