A good presenter will share their slide deck on the web after their presentation. But many times the slides are only a shell of the real talk. Fortunately, with HTML5 audio, we can add our voice back to our slides and recreate the real presentation.
// TODO update link to html5hacks site To show what we're trying to accomplish, I've created a very basic slide deck with audio narration which briefly describes the issue at hand.
Back when the web was just taking off, it was common (bad) practice to include audio on your page. I'm not talking about a Flash-based music player, but rather the more primitive audio solution: <bgsound>
. Those who were programming back when HTML 3.2 came out will be familiar with this oft-forgotten tag.
Luckily for us, <bgsound>
isn't the end of the story. According to the latest W3C spec, <bgsound>
has a much friendlier HTML5 alternative that you've likely heard of: the <audio>
tag.
So what benefits does <audio>
bring us? Well, <bgsound>
was an IE only property. <audio>
on the other hand has wide support, with only IE 7 & 8 lacking functionality. <audio>
also allows API access so that we can control playback, seek through the audio, and even manipulate the data with the MediaStream API. Plus, the <audio>
tag allows you to use native controls and/or provide your own customized controls.
Before getting in to the details on how we're going to use the <audio>
tag, we need to talk a little about file formats. The MP3 format has gained tremendous popularity over the last decade and a half, but unfortunately due to licensing requirements, relying on MP3's for our audio is a messy situation.
Luckily for us, the <audio>
tag supports multiple formats gracefully. This means we can create a patchwork of audio file formats to gain full browser support. And we'll need a patch work because no one format is currently supported across all browsers.
For our needs, we've created two files: an MP4/AAC file and an OggVorbis file.
If you'd like to read more on the subject, I highly recommend Ashley Gullen's post 'On HTML5 audio formats - AAC and Ogg'.
We can load our audio files by adding in two <source>
tags with information about our two audio files inside of the <audio>
tag:
<audio controls id="myPlayer">
<source src="myAudio.m4a" type="audio/mp4" />
<source src="myAudio.ogg" type="audio/ogg" />
Your browser does not support HTML5 audio.
</audio>
There are two attributes for each <source>
tag. The 'src' attribute, whose value is the path to the audio file, and the 'type' attribute, whose value is the MIME type of the file.
Again, the browser will choose whichever file it supports without you having to do any detective work. Very nice.
Okay, so now if we load this into a webpage we'll get a simple audio player that we can manually control. What's nice is that since we used the 'controls' attribute, the audio player controls are built for us by the browser. This makes allowing manual control of our audio very simple.
For our needs, we want to control the playback of the audio programmatically. To do this, let's take a look at the API for starting and stopping playback. The element has two built-in methods for this, 'play' and 'pause'. Calling those methods is straightforward:
var audioPlayer = document.getElementById('myPlayer');
audioPlayer.play();
audioPlayer.pause();
These methods will come in handy in a moment when we want to start playing our audio after we change slides.
The other part of the equation is the ability to seek to different locations in our audio. Again, this is very simple. Our element has a 'currentTime' property that can be both get and set (in seconds).
console.log(audioPlayer.currentTime); // returns 0 since we haven't started playing the audio yet
audioPlayer.currentTime = 10; // move to 10 seconds into the audio
console.log(audioPlayer.currentTime); // returns 10
audioPlayer.play();
setTimeout(function () {
console.log(audioPlayer.currentTime); // returns 11
}, 1000);
As you can see, getting and setting the current time is a trivial process. In the Part 2, we'll put this functionality to use by adding narration to slides.
So now we've got the building blocks for implementing a slide narrator. To make things easier, we're going to use the fantastic 'Deck.js' project as our HTML slide framework. Deck.js supports extensions, which allows you to add functionality to your slides beyond what's already provided.
We'll be creating a new extension called Narrator. For brevity's sake, I won't get into the details of Deck.js or creating extensions, but you can check out the code in the deck.narrator.js GitHub repo.
Our extension boils down to one requirement: It should automatically play a defined section of audio on each slide.
That might sound simple, but we need to figure out a couple of things first:
- How will we define what audio to play for each slide?
- How will we stop the audio after it gets to the end of the section
There are a couple of ways to define what segment of the audio each slide plays. You could define a start time and a stop time for each slide, but that seems like too much work. Instead we'll just define how long each slide should play for, and then calculate the implied start and stop timestamps for each slide.
To store our audio duration, we'll take advantage of HTML5 data-* attributes by creating a custom 'data-narrator-duration' attribute. The value of this will be the number of seconds to play the audio for. Here's a sample slide element for a Deck.js HTML file.
<section class="slide" data-narrator-duration="2">
...
</section>
Then, upon page initialization, we'll loop through each slide element and calculate the proper start/stop timestamps for each slide. This is important in case our viewer wants to move through the slides in a non-linear fashion. Here's the basic code:
// create an array for our segment timestamps
var segments = [];
// create a placeholder for our audio element reference
var audio;
// we'll get to this variable later
var segmentEnd = 0;
function init () {
// get the audio element we added to our page
audio = document.getElementById('audioNarration');
// use deck.js built-in functionality to get all slides and current slide
var $slides = $.deck('getSlides');
var $currentSlide = $.deck('getSlide');
// set initial values for time position and index
var position = 0;
var currentIndex = 0;
// now loop through each slide
$.each($slides, function(i, $el) {
// get the duration specified from the HTML element
var duration = $el.data('narrator-duration');
// this determines which slide the viewer loaded the page on
if ($currentSlide == $el) {
currentIndex = i;
}
// push the start time (previous position) and end time (position + duration) to an array of slides
segments.push([position, position + duration]);
// increment the position to the start of the next slide
position += duration;
});
}
Now that we've got our segment timestamps defined, let's look at playing that audio on each slide transition. Deck.js fires a 'deck.change' event when the slide is changed, so we can hook into that and have it call our changeSlides function, which looks like this:
function changeSlides (e, from, to) {
// check to make sure audio element has been found
if(audio) {
// move the playback to our slides start
audio.currentTime = segments[to][0];
// define the end of our section
segmentEnd = segments[to][1];
}
}
Most of the code makes sense, but I do want to talk about the 'segmentEnd' line and what it's doing.
Unfortunately, you can't give the play()
function an amount of time to play for. Once you start playing, it will keep going until it runs out of audio or you tell it to pause. Thankfully, the audio element emits a 'timeupdate' event which we can listen to in order to pause playback once our segment timestamp has been reached. We can add that listener just like any other event listener:
audio.addEventListener('timeupdate', checkTime, false);
Our 'checkTime' function is very small. All it does is check to see if currentTime in the audio is greater than the segmentEnd time. If so, it pauses our audio:
function checkTime () {
if (audio.currentTime >= segmentEnd) {
audio.pause();
}
}
Now that we've got our audio hooked up to our slides, we can take advantage of the other extensions already written for Deck.js. https://github.com/rchampourlier/deck.automatic.js/ is an extension that makes your slides run automatically. By including this extension with our presentation, we can recreate that 'presentation' feel to our slides.
Aside from the going through the steps of adding the automatic extension, we also need to make sure that if a user starts/stops the audio, we start/stop the slideshow playback. To do this, we'll sync up the 'play' and 'pause' events of our audio element with the automatic extension. For simplicity, we're going to control all slide playback using our audio controls and leave off the deck.automatic.js playback control.
Deck.automatic.js adds some events to the mix, including 'play' and 'pause' events. By triggering these events when our similarly named audio events fire, we can make sure our slides are in sync with our content.
We add two simple functions to our extension:
function startSlides (ev) {
$.deck('play');
}
function stopSlides (ev) {
$.deck('pause');
}
And then add our event listeners in the deck.init callback:
$d.bind('deck.init', function() {
// ... other code here ...
// Sync audio with slides
audio.addEventListener('play', startSlides, false);
audio.addEventListener('pause', stopSlides, false);
});
Also, since our slides automatically advance, we need to comment out the 'timeupdate' event listener which would pause our audio at the end of a slide.
With those things taken care of, our slides and audio automatically transition to create a seamless experience.
One thing the code doesn't take into consideration is if the user navigation the audio themselves. We could add this by listening to the 'seeked' event from the audio element and calculating where in the slide deck we should move to. For now, we're not going to worry about it.
That's the majority of the code. I left out a few details in relation to some deck.js configurations, so again check out the GitHub repo for the full example.