Everybody dance now
with Web Audio API
Jaume Sanchez @thespite
http://www.clicktorelease.com
Audio on the browser
Flash and other plugins
HTML5 Audio Element
Hybrid approach
http://www.schillmania.com/projects/soundmanager2/
Web Audio API
(and a bit of Mozilla Audio Data API)
The Web Audio API allows developers to manipulate
and play audio assets on a Web page or application.
"Web Audio API is to HTML5 <audio>
what canvas is to <img>"
Mozilla Audio Data API tried with a
JavaScript-centric approach.
How to use
var context = new AudioContext();
Usually you'll want one AudioContext (AC) per page.
Once there's an AC, you can start
adding nodes and connecting them.
Nodes
(nodes everywhere)
Modular routing.
Basic audio operations are performed with audio nodes.
Audio nodes are linked together to form audio routing graphs.
Audio nodes are linked via their inputs and outputs.
This modular design allows the flexibility to create complex
audio functions with dynamic effects.
Routing graphs
Sources
We need some element to input data into the system.
(something has to make noise)
createBufferSource
createMediaElementSource
createMediaStreamSource
createOscillator
var context = new AudioContext();
var source = context.createBufferSource();
source.buffer = buffer;
source.start();
Destinations
Any other node, or the mixer.
The graph usually ends with context.destination
var context = new AudioContext();
var source = context.createBufferSource();
source.buffer = buffer;
source.connect( context.destination );
source.start();
but the combinations are endless
oscillator.connect( gain );
gain.connect( convolver );
convolver.connect( notchFilter );
notchFilter.connect( context.destination );
Buffer Source
Creates an AudioBufferSourceNode interface.
context.createBufferSource();
buffer attribute: IEEE 32 bit float ranging -1 to 1
playbackRate attribute: AudioParam
loop/loopStart/loopEnd attributes
start() and stop() methods
onended event
Buffer Source demo
var context = new AudioContext();
var request = new XMLHttpRequest();
request.open( 'GET', 'sound.mp3', true );
request.responseType = 'arraybuffer';
request.onload = function() {
context.decodeAudioData( request.response, function( buffer ) {
var source = context.createBufferSource();
source.buffer = buffer;
source.connect( context.destination );
source.start();
}, function() {
console.log( 'error' );
} );
};
request.send();
Load Sound Demo
Load Sound reusing AudioBuffer Demo
Analyser
Creates an AnalyserNode to provide real-time
frequency and time-domain analysis information
context.createAnalyser()
fftSize attribute: size of FFT - power of two
getFloatFrequencyData: copies frequency-domain data
getByteFrequencyData: copies frequency-domain data
getByteTimeDomainData: copies time-domain data
Load Sound and show FFT Demo
getUserMedia Demo
AudioParam
float value (and defaultValue)
Allows scheduling of value changes
setValueAtTime
linearRampToValueAtTime
exponentialRampToValueAtTime
setTargetAtTime
setValueCurveAtTime
cancelScheduledValues
Gain
Create a GainNode to control the volume.
context.createGain();
gain attribute: AudioParam
var context = new AudioContext();
var source = context.createBufferSource();
source.buffer = buffer;
var volume = context.createGain();
volume.gain.value = .1;
source.connect( volume );
volume.connect( context.destination );
source.start();
Delay
Creates a DelayNode to add delay.
context.createDelay( maxDelayTime );
delayTime attribute AudioParam
var context = new AudioContext();
var source = context.createBufferSource();
source.buffer = buffer;
var delay = context.createDelay( 100 );
delay.delayTime.value = 3;
source.connect( delay );
delay.connect( context.destination );
source.start();
Delay + Gain = Reverb
Creating a loop between a gain and a delay node
creates a reverb effect on the stream.
var context = new AudioContext();
var a = document.createElement( 'audio' );
a.src = 'track.wav';
a.loop = false;
var audioSource = context.createMediaElementSource( a );
var delayNode = context.createDelay( 1 );
var gainNode = context.createGain();
gainNode.gain.value = .5;
delayNode.delayTime.value = .1;
audioSource.connect( delayNode );
delayNode.connect( gainNode );
gainNode.connect( delayNode );
delayNode.connect( context.destination );
a.play();
Gain + Delay Demo
Oscillator Source
Creates an OscillatorNode.
context.createOscillator();
type attribute:
SINE, SQUARE, SAWTOOTH,
TRIANGLE, CUSTOM
setWaveTable() method for CUSTOM type
frequency attribute AudioParam
var context = new AudioContext();
var oscillator = context.createOscillator();
oscillator.frequency.value = 440;
oscillator.type = 4;
oscillator.connect(context.destination);
oscillator.start();
Filters!
Creates a BiquadFilterNode that provides several
default stock second order filters.
context.createBiquadFilter();
type attribute
LOWPASS, HIGHPASS, BANDPASS,
LOWSHELF, HIGHSHELF, PEAKING,
NOTCH, ALLPASS
frequency attribute AudioParam
Q attribute AudioParam
gain attribute AudioParam
Oscillator and Biquad Filter Demo
Convolution (euh!)
Creates a ConvolverNode for linear convolution effects.
context.createConvolver();
buffer attribute: AudioBuffer with impulse response.
Convolution Demo
Panner
Creates a PannerNode for spatialisation.
context.createPanner();
Defines:
- position, velocity, direction of sample
- distance model
- directional sound cone
Tied to AudioListener (context.listener) for
spatial sound and doppler shift effect.
Other
createChannelMerger
createChannelSplitter
createDynamicsCompressor
createPeriodicWave
createWaveShaper
createJavaScriptNode
All together
There's a main ambient sound loop with gain.
The camera has a sound loop with gain.
Each cube has:
- a basic sound loop with gain
- a proximity sound loop with gain
- an activation sound with gain
- a deactivation sound with gain
- a sound analyser
- a panner
All sounds are merged in a convolver, a filter, a gain
and a dynamics compressor
Awesome.
Now what?
WebRTC getUserMedia
Web MIDI API
Procedural sound generation
Real-time effects
References
The spec
Getting Started with Web Audio API | HTML5 Rocks
Mixing Positional Audio and WebGL | HTML5 Rocks
Thanks!
Questions?
Let's talk!
Jaume Sanchez @thespite
http://www.clicktorelease.com
https://github.com/spite
Barcelona Future JS 2014
By Jaume Sánchez
Barcelona Future JS 2014
- 4,712