🔥 HTML Elements
In this post on Astro JS Mux Video, we see how you can integrate the Mux HTML components video player into your site. We will also see that you can put a site together quite quickly with Astro. And that is without the need for learning frameworks like React and Svelte. Astro is principally a static site builder, thought it also support server rendered content. It is aligned with the islands architecture philosophy. That is where components on a page can exist and hydrate independently of each other. This is a recipe for fast sites.
Mux is a video streaming service. You upload your video in a single format and they sweat the details of making it work with user devices, be they Android or iOS mobile, Windows, Linux or MacOS desktop. In fact, you can even have the player stream to Chromecast compatible devices. The Mux Player is a custom element or Web Component. The advantage of using Web Components is that, in theory, you can write your components once and use them in any web framework or with none at all, just like we will today!
There is minimal setup and you really can build out an Astro site using the Mux Custom Elements video player in a matter of minutes. So let’s punch it!
🧱 Getting Started with Mux Video Player
You do not need a Mux account if you want to follow along. As an aside, however, we see how to set one up in the Svelte Video Blog Tutorial. To play a Mux video, you just need a Mux playback ID for the video. We have a Mux promotional asset ID which we can use here for this purpose.
If you are following along, let’s start by spinning up a new Astro project from the Terminal:
pnpm create astro@latest astro-mux-video
cd astro-mux-video
pnpm install
pnpm astro telemetry disable
pnpm install @mux/mux-video
pnpm run dev
At the time of writing the Mux player is still in beta testing. Taking that into account, you should probably avoid using this setup for production projects.
In the block above we set up a new Astro project, disabled Astro telemetry (which is enabled by default) and installed the @mux/mux-video
package in our project. Ignore the telemetry disable
step if you are OK with sending data to Astro.
📀 Video Component
The Mux Player is a Web Component. All we need to do to make it work with Astro is export the NPM package in a JavaScript (or TypeScript) file. The final step, to make sure it gets bundled with our site, is then to add a link to this file in a script tag. We will do that on our home page which we will look at next.
Here’s the code for the Video Component. Make a src/components
folder and create a mux-video.ts
file in there:
export * as MuxVideo from '@mux/mux-video';
Hope that wasn’t too much typing 😅.
📼 Home Page
Essentially, as we just mentioned, you now just need to import the component in a script tag, then add a mux-video
element in the Astro markup. Here is a pared back version:
---
---
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<script>import '../components/mux-video.ts'</script>
<script defer src="https://www.gstatic.com/cv/js/sender/v1/cast_sender.js?loadCastFramework=1"></script>
<title>Mux Video Player in 🖖🏽 Astro</title>
</head>
<body>
<main class="container">
<mux-video playback-id="EcHgOK9coz5K4rjSwOkoE7Y7O01201YMIC200RI6lNxnhs" controls />
</main>
</body>
</html>
In line 9
we make the import mentioned. The script
tag in line 10
, meanwhile adds Chromecast support — it’s that simple! The player also supports Airplay (used by Apple devices). We use minimal attributes on the mux-video
element. You see this web component does not look too different to a regular HTML component. In fact we will see below, we can add the usual attributes we have on an HTML5 video element. We use a Mux promotional video for the playback-id here. When using your own video, You’ll notice Mux generate two IDs. It is the public playback-id rather than the master-access ID which you will need here.
💫 Levelling Up
In the markup below, we take it up a level, adding attributes to reduce layout shift and video controls as well as some styling. We learn more about reducing cumulative shift and lazy-loading video in the Svelte Video blog tutorial. We use SvelteKit there, though Astro works with Svelte components.
---
---
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<link rel="icon" href="/favicon.ico" sizes="any" />
<link rel="icon" href="/icon.svg" type="image/svg+xml" />
<link rel="apple-touch-icon" href="/apple-touch-icon.png" />
<link rel="manifest" href="/manifest.webmanifest" />
<script>import '../components/mux-video.ts'</script>
<script defer src="https://www.gstatic.com/cv/js/sender/v1/cast_sender.js?loadCastFramework=1"></script>
<title>Mux Video Player in 🖖🏽 Astro</title>
</head>
<body>
<main class="container">
<h1 class="heading">Mux Video Player in 🖖🏽 Astro</h1>
<figure>
<div class="video-container">
<mux-video playback-id="EcHgOK9coz5K4rjSwOkoE7Y7O01201YMIC200RI6lNxnhs" autoplay controls loop width="688px"
height="387px" poster="/mux-video-player-poster.png" preload="none" primary-color="#31274a"
secondary-color="#ff5d01">
</mux-video>
</div>
<figcaption>VIDEO CREDIT: William Ehrendreich: <a href="https://www.videvo.net/profile/williamehrendreich/"
rel="nofollow noopener noreferrer">www.videvo.net/profile/williamehrendreich/</a> </figcaption>
</figure>
<p>Boldly going where no video player has gone before…</p>
</main>
</body>
</html>
<style>
/* space-grotesk-500 - latin */
@font-face {
font-family: Space Grotesk;
font-style: normal;
font-weight: 500;
src: local(''),
url('/fonts/space-grotesk-v12-latin-500.woff2') format('woff2'),
/* Chrome 26+, Opera 23+, Firefox 39+ */
url('/fonts/space-grotesk-v12-latin-500.woff') format('woff');
/* Chrome 6+, Firefox 3.6+, IE 9+, Safari 5.1+ */
}
/* space-grotesk-700 - latin */
@font-face {
font-family: Space Grotesk;
font-style: normal;
font-weight: bold;
src: local(''),
url('/fonts/space-grotesk-v12-latin-700.woff2') format('woff2'),
/* Chrome 26+, Opera 23+, Firefox 39+ */
url('/fonts/space-grotesk-v12-latin-700.woff') format('woff');
/* Chrome 6+, Firefox 3.6+, IE 9+, Safari 5.1+ */
}
:root {
/* mercury */
--colour-light: hsl(0deg 5% 92%);
/* green spring */
--colour-alt: hsl(107deg 7% 74%);
/* Xiketic */
--colour-dark: hsl(240deg 100% 4%);
/* BlazeOrange */
--colour-theme: hsl(22 100% 50%);
}
body {
margin: var(--spacing-0);
min-height: 100vh;
--spacing-0: 0;
--spacing-1: 0.25rem;
--spacing-2: 0.5rem;
--spacing-12: 3rem;
--spacing-18: 4.5rem;
--max-width-wrapper: 48rem;
--font-size-root: 16px;
--font-size-1: 1rem;
--font-size-3: 1.563rem;
--font-size-6: 3.052rem;
--font-weight-medium: 500;
--font-weight-bold: 700;
font-family: Space Grotesk;
font-weight: var(--font-weight-medium);
background-color: var(--colour-dark);
/* CREDIT https://www.joshwcomeau.com/gradient-generator/ */
background-image: linear-gradient(45deg,
hsl(240deg 100% 4%) 0%,
hsl(243deg 37% 10%) 27%,
hsl(246deg 32% 15%) 41%,
hsl(253deg 31% 20%) 53%,
hsl(285deg 36% 26%) 64%,
hsl(327deg 60% 38%) 75%,
hsl(347deg 72% 51%) 86%,
hsl(22deg 100% 50%) 99%);
}
a {
color: var(--colour-theme)
}
.container {
color: var(--colour-alt);
width: min(100% - var(--spacing-12), var(--max-width-wrapper));
margin: var(--spacing-18) auto;
font-size: var(--font-size-3);
text-align: right;
}
.heading {
color: var(--colour-light);
text-align: center;
font-size: var(--font-size-6);
font-weight: var(--font-weight-bold);
margin-bottom: var(--spacing-18)
}
mux-video {
background-color: var(--colour-theme);
}
.video-container,
mux-video {
max-width: var(--max-width-full);
aspect-ratio: 16 / 9;
}
figure {
margin-bottom: var(--spacing-12);
text-align: left;
}
figcaption {
font-size: var(--font-size-1);
margin-block: var(--spacing-2);
}
</style>
I added self-hosted fonts here. For this to work, you need to download them from google-webfonts-helper, then extract the zip and place the .woff
and .woff2
files in a new public/fonts
folder in your project.
There is a live demo if you want to see the full thing in action.
🙌🏽 Astro JS Mux Video: Wrapping Up
We have taken a look at the Mux custom elements video player in this post. In particular, we saw:
- how to add the mux-video to an Astro site,
- linking your Mux video assets in the mux-video player,
- that Astro can work without a framework like React or Svelte.
We have just scratched the surface of what we can do with the Mux custom elements player in this Astro JS Mux video post. The video works fine on this simple page, where it is the main content. However, for video lower down the page, you might consider using a Svelte or React component. Svelte, especially, can make it easy to add lazy loading. This is important for making your page load faster. If you prefer to stay frameworkless, many modern browsers have lazy loading support and you can access this adding a loading="lazy"
attribute. Also check out Mux docs for more customisations.
Take a look at the full project code on the Rodney Lab GitHub page. I hope you found this article useful and am keen to hear how you will the starter on your own projects as well as possible improvements.
🙏🏽 Astro JS Mux Video: Feedback
Have you found the post useful? Would you prefer to see posts on another topic instead? Get in touch with ideas for new posts. Also if you like my writing style, get in touch if I can write some posts for your company site on a consultancy basis. Read on to find ways to get in touch, further below. If you want to support posts similar to this one and can spare a few dollars, euros or pounds, please consider supporting me through Buy me a Coffee.
Finally, feel free to share the post on your social media accounts for all your followers who will find it useful. As well as leaving a comment below, you can get in touch via @askRodney on Twitter and also askRodney on Telegram. Also, see further ways to get in touch with Rodney Lab. I post regularly on Astro as well as SvelteKit. Also subscribe to the newsletter to keep up-to-date with our latest projects.