This package has two indeterminate progress spinners, one with lines:
…and the other with dots:
They are two of the spinners from Ionic Framework.
Because sometimes you just need a simple spinner without a 5-tonne framework attached.
When including a spinner in your site or app, the element that contains your spinner should be set as an ARIA live region and provide meaningful, app-specific feedback on its current state.
Learn more: “It’s alive!”: Apps That Feed Back Accessibly, More Accessible Skeletons
If you just want the SVGs, you can grab them below (right click and save):
<svg stroke="currentColor" height="1em" viewBox="0 0 64 64"><g stroke-width="4" stroke-linecap="round"><line y1="12" y2="20" transform="translate(32,32) rotate(180)"><animate attributeName="stroke-opacity" dur="750ms" values="1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0;1" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(210)"><animate attributeName="stroke-opacity" dur="750ms" values="0;1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(240)"><animate attributeName="stroke-opacity" dur="750ms" values=".1;0;1;.85;.7;.65;.55;.45;.35;.25;.15;.1" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(270)"><animate attributeName="stroke-opacity" dur="750ms" values=".15;.1;0;1;.85;.7;.65;.55;.45;.35;.25;.15" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(300)"><animate attributeName="stroke-opacity" dur="750ms" values=".25;.15;.1;0;1;.85;.7;.65;.55;.45;.35;.25" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(330)"><animate attributeName="stroke-opacity" dur="750ms" values=".35;.25;.15;.1;0;1;.85;.7;.65;.55;.45;.35" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(0)"><animate attributeName="stroke-opacity" dur="750ms" values=".45;.35;.25;.15;.1;0;1;.85;.7;.65;.55;.45" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(30)"><animate attributeName="stroke-opacity" dur="750ms" values=".55;.45;.35;.25;.15;.1;0;1;.85;.7;.65;.55" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(60)"><animate attributeName="stroke-opacity" dur="750ms" values=".65;.55;.45;.35;.25;.15;.1;0;1;.85;.7;.65" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(90)"><animate attributeName="stroke-opacity" dur="750ms" values=".7;.65;.55;.45;.35;.25;.15;.1;0;1;.85;.7" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(120)"><animate attributeName="stroke-opacity" dur="750ms" values=".85;.7;.65;.55;.45;.35;.25;.15;.1;0;1;.85" repeatCount="indefinite"></animate></line><line y1="12" y2="20" transform="translate(32,32) rotate(150)"><animate attributeName="stroke-opacity" dur="750ms" values="1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0;1" repeatCount="indefinite"></animate></line></g></svg>
<svg height="1em" fill="currentColor" viewBox="0 0 64 64"><g><circle cx="16" cy="32" stroke-width="0"><animate attributeName="fill-opacity" dur="750ms" values=".5;.6;.8;1;.8;.6;.5;.5" repeatCount="indefinite"></animate><animate attributeName="r" dur="750ms" values="3;3;4;5;6;5;4;3" repeatCount="indefinite"></animate></circle><circle cx="32" cy="32" stroke-width="0"><animate attributeName="fill-opacity" dur="750ms" values=".5;.5;.6;.8;1;.8;.6;.5" repeatCount="indefinite"></animate><animate attributeName="r" dur="750ms" values="4;3;3;4;5;6;5;4" repeatCount="indefinite"></animate></circle><circle cx="48" cy="32" stroke-width="0"><animate attributeName="fill-opacity" dur="750ms" values=".6;.5;.5;.6;.8;1;.8;.6" repeatCount="indefinite"></animate><animate attributeName="r" dur="750ms" values="5;4;3;3;4;5;6;5" repeatCount="indefinite"></animate></circle></g></svg>
You can also use the spinners via plain old JavaScript:
require()
or import()
the spinner you want in your project.
Instantiate the class and pass it an options object that contains the DOM element that you want the class to bind to any properties you want to include. Valid properties are:
size
: the size of the spinner (height; width is auto-calculated). Default: 1em
. You can use any CSS sizing unit.colour
: the colour of the spinner (default: uses the colour of the surrounding context). You can use any valid CSS colour value.show
: a boolean indicating whether the spinner is visible. Default: true
.<main>
<div id='spinner'></div>
<script type='module'>
import { Lines } from 'https://unpkg.com/@small-tech/spinners@1.0.0/dist/index.mjs'
const lines = new Lines({
target: document.getElementById('spinner'),
props: {
size: '2em',
colour: 'SlateGrey'
}
})
</script>
</main>
You can find a version of this example in the examples/javascript folder.
Finally, you can also use the spinners in your Svelte projects.
The example below is functionally identical to the plain old JavaScript one, above.
<script>
import { Lines } from '@small-tech/spinners'
</script>
<Lines size='2em' colour='SlateGrey' />
You can find a version of this example in the examples/svelte folder.
In an ideal world, animated SVGs (yes, even those that use SMIL animations – which are great, by the way) should be supported everywhere. Sadly, we don’t live in an ideal world. We live in a world where GitHub can mass-violate the copyright of free and open source projects implement “AI pair programming” but cannot render a simple animated SVG properly.
So, if you’re viewing this readme on GitHub, what you’re seeing above are bitmap versions (animated PNGs) of the spinners exported using tiny scripts I wrote for that purpose. The reason they’re a shade of blue is because, unlike the SVGs which default to using the currentColor
of their context, I had to export a colour that would work with acceptable contrast on both light and dark mode.
They’re embedded as fallbacks within <foreignObject>
tags inside of the inline SVGs in the readme (so, for example, if you’re viewing them in VSCodium, you will only see the SVG spinners, not two sets of spinners).
You can also use these scripts to generate your own animated PNG versions of the spinners, should you so desire.
First, clone this repository. Then, install the dependencies:
cd png
npm install
cd ..
And run the scripts:
node png/lines
node png/dots
You can customise the size, colour, and frame multiplier (animation resolution) via options.
To see all options:
node png/lines --help
node png/dots --help
If you’re interested in the nitty gritties of image manipulation, view the source of the lines.js and dots.js scripts to see an example of how to convert SVGs to animated PNGs in Node.js using the svg-png-converter and node-apng modules.
(Yes, I know I should refactor the redundancies between those two scripts. No, I don’t have the time to do that now. And yes, I’ve already spent far more time than on this than any reasonable person should have.)
Small Technology Foundation is a tiny, independent not-for-profit.
We exist in part thanks to patronage by people like you. If you share our vision and want to support our work, please become a patron or donate to us today and help us continue to exist.