⏳ The Loading Indicator
The <ProgressBar />
and <LoadingIndicator />
components offer loading indicators for your view transitions. To see them in action, visit the ProgressBar page or the LoadingIndicator page and press the “Go to page two” link.
There is also a demo for a crazily styled <ProgressBar />
here, which can be started by pressing on “Showtime”.
Contents
- Why should I use this component?
- How does it work?
- Usage
- Alternative 1: Use it as an Astro integration
- Alternative 2: Use the <LoadingIndicator /> or the <ProgressBar /> component
- Styling the <ProgressBar /> with CSS
- Styling the <LoadingIndicator /> with CSS
- Use your own #vtbot-loading-indicator
- Using CSS to define your custom loading indicator
- Using JavaScript to define your custom loading indicator
- Using custom loading indicators
- <BrakePad/>: Slowing things down during development 🦥
- Publish or contribute your custom loading indicator! (Pocycli!)
Why should I use this component?
The loading indicator is a great way to let your users know that something is happening. For view transitions, this is even more important than for normal page loads, as the view transition animation only starts after the target page has fully loaded. With a slow internet connection or heavy server-side rendering, this can take quite a while. From the instant the user clicked the link up to the start of the view transition animation the user typically gets no visual feedback. This gap is now filled by the loading indicator.
How does it work?
The Bag offers very versatile support for loading indicators. The common feature is to start an action with the click on the link and stop it again when the target page is fully loaded.
To achieve this, <LoadingIndicator/>
inserts a loading
class into the html element of the page and removes it later. You can also have it call your Javascript code to show and hide something.
Without any further configuration, the LoadingIndicator component displays a pulsating version of your favicon in the top right corner of the screen during loading.
Usage
In good Astro tradition this component works out of the box without any configuration. But if you want you can also customize it completely to your needs.
Alternative 1: Use it as an Astro integration
The easiest way to enjoy the loading indicator is to install astro-vtbot as an Astro integration. This will automatically add the <LoadingIndicator />
component to all pages that use the <ClientRouter />
component.
If you want to keep the integration, e.g. for the automatic linter support, but want to (temporarily) disable the loading indicator, set loadingIndicator to false in the configuration:
Favicon image
You should have a <link rel="icon" href="/favicon.xyz">
in your page <head>
element. This will be used as the image for the loading indicator. If you do not have a favicon link, <LoadingIndicator>
will try to load the standard /favicon.ico
. If that is not there, it will show a grey circle as loading indicator.
<LoadingIndicator/>
supports various image formats but it works best with raster images with a minimal resolution of 128 x 128 or with .svg
files. It works not so well with small .ico
files.
The image is contained in a <div>
element which has the id vtbot-loading-indicator
and you might apply your own styling to it. The generates code will look like this
or
Alternative 2: Use the <LoadingIndicator /> or the <ProgressBar /> component
If you do not install astro-vtbot
as an Astro integration, or if you set loadingIndicator
to false in your Astro configuration, you can still install the <LoadingIndicator />
component directly on your page. The <LoadingIndicator />
component should be put inside the <head>
element behind the <ClientRouter />
component.
Without further configuration, the <LoadingIndicator />
component will display a pulsating version of your favicon in the top right corner of the screen during loading.
You can configure the source of the image as well as the position. These is the interface:
The src
attribute is a href, the position strings are values for the corresponding CSS attributes, e.g.
As long as you include the <LoadingIndicator />
component behind the <ClientRouter />
in the <head>
its attributes will override the defaults of the <LoadingIndicator />
component that is installed by the Astro integration. So it is ok to have the default via integration on most pages and override it on some pages with an explicit <LoadingIndicator>
element.
The <ProgressBar />
is used the same way.
There are no properties yet to control the behavior of the <ProgressBar />
. But you can style its appearance wih CSS.
Styling the <ProgressBar /> with CSS
The <ProgressBar />
component is implemented using Swup’s progress bar plugin.
Thus styling is straight forward:
Styling the <LoadingIndicator /> with CSS
As soon as the user clicks the link, the CSS class loading
is added to the body element of the page. Once the target page is loaded completely and the view transition is about to begin, the loading
CSS class is removed again. This allows you to customize the loading indicator via CSS.
The default CSS covers positioning, sizing, and pulsing of the indicator.
You can override it with your own styling. Best way to do this is to define your own custom loading indicator.
Trick: The animation defined above runs a one second cycle over and over. It keeps the loading indicator invisible for the first 30% of the cycle. I.e. for the first 300 ms your indicator does not show. Or in other words: Even though the
.loading
class is added immediately after the user clicks the link, the indicator will not appear for the first 300 ms. This is a good thing, as it only shows the indicator for slow loading pages and not for a page that is already in the browser cache.
Use your own #vtbot-loading-indicator
If you want to use your own loading indicator element, you can do so by adding an HTML element with the ID vtbot-loading-indicator
to your page. In this case the <LoadingIndicator/>
component does not insert its own loading indicator. Again, best way to do this is to define your own custom loading indicator.
Using CSS to define your custom loading indicator
You can define a custom loading indicator by extending the <LoadingIndicator />
component.
Typically, your component will define an HTML element and styling.
The logic for showing and hiding the loading indicator will be handled by the <LoadingIndicator />
component.
Here is the template that you also find in the astro-vtbot
package.
If you are happy with the default loading indicator html you can keep it. Otherwise you define your own, see line 10 above and the section on alternative elements.
Using JavaScript to define your custom loading indicator
You do not want to craft animations solely with CSS and prefer to use some JavaScript library like
GSAP et.al? No problem. Also in that case you can craft your custom loading indicator. Instead of <LoadingIndicator/>
component use the loading-indicator.ts
script.
Here is the template that you also find in the astro-vtbot
package.
You just define the two functions show and hide and tell the loading-indicator
about them. In the body of these two functions you can craft your special effects!
Using custom loading indicators
Your custom loading indicator components can then be used instead of <LoadingIndicator>
. No need anymore for the <LoadingIndicator />
component in the <head>
. Whether your custom component needs to be inserted into the head or the body depends on the HTML elements the template section of your component directly inserts into the DOM:
- No elements at all? Your component can go either to the
<head>
or to the<body>
<meta>
or other<head>
stuff? Your component goes into the<head>
<div>
or other<body>
stuff? Your component goes into the<body>
- Both? No good idea. Maybe you can generate the body part dynamically?
<BrakePad/>: Slowing things down during development 🦥
Testing loading indicators is hard with your slick local server. Transitions happen in no time and you can not even get a glimpse of your flashy indicator? You can use your browser’s developer tools to simulate a slow network connection!
But you can also put the brakes on, with the <BrakePad/> from the Bag of Tricks.
Publish or contribute your custom loading indicator! (Pocycli!)
I’m curious to see what you will build!
If you define your own custom loading indicator - either by extending <LoadingIndicator/>
or by defining show
& hide
with loading-indicator.ts
as described above: Please drop me a line (issue or pull request at https://github.com/martrapp/astro-vtbot-website↗, or DM @Martin at Astro’s Discord server) and I will gladly link to your extension or you add it as a contribution to the 👜 Bag of Tricks. ✨