<html>
<head>
<meta charset="utf-8">
<title>JS Bin</title>
</head>
<body>
<button id="shot1Normal">Shot 1</button>
<button id="shot2Normal">Shot 2</button>
<p>
<button id="shot1Repeated">Shot 1 repeated</button>
<button id="shot2Repeated">Shot 2 repeated</button>
<p>
<button id="shot1RepeatedIntervalRandom">Shot 1 repeated at random intervals</button>
<button id="shot2RepeatedPitchAndIntervalRandom">Shot 2 repeated, pitch and interval random</button>
<p>
<canvas id="myCanvas" width=300 height=100></canvas>
<div class="controls">
</body>
</html>
var bufferLoader;
var ctx;
var canvas;
var canvasContext;
var gradient;
var analyser, analyserLeft, analyserRight;
var width, height;
var dataArray, bufferLength;
var dataArrayLeft, dataArrayRight;
var listOfSoundSamplesURLs = [
'https://mainline.i3s.unice.fr/mooc/shoot1.mp3',
'https://mainline.i3s.unice.fr/mooc/shoot2.mp3'
];
window.onload = function init() {
// To make it work even on browsers like Safari, that still
// do not recognize the non prefixed version of AudioContext
var audioContext = window.AudioContext || window.webkitAudioContext;
ctx = new audioContext();
loadAllSoundSamples();
createStaticNodes();
canvas = document.querySelector("#myCanvas");
width = canvas.width;
height = canvas.height;
canvasContext = canvas.getContext('2d');
// create a vertical gradient of the height of the canvas
gradient = canvasContext.createLinearGradient(0,0,0, height);
gradient.addColorStop(1,'#000000');
gradient.addColorStop(0.75,'#ff0000');
gradient.addColorStop(0.25,'#ffff00');
gradient.addColorStop(0,'#ffffff');
requestAnimationFrame(visualize);
};
function visualize() {
clearCanvas();
drawVolumeMeters();
drawWaveform();
// call again the visualize function at 60 frames/s
requestAnimationFrame(visualize);
}
function drawWaveform() {
canvasContext.save();
// Get the analyser data
analyser.getByteTimeDomainData(dataArray);
canvasContext.lineWidth = 2;
canvasContext.strokeStyle = 'lightBlue';
// all the waveform is in one single path, first let's
// clear any previous path that could be in the buffer
canvasContext.beginPath();
var sliceWidth = width / bufferLength;
var x = 0;
// values go from 0 to 256 and the canvas heigt is 100. Let's rescale
// before drawing. This is the scale factor
heightScale = height/128;
for(var i = 0; i < bufferLength; i++) {
// dataArray[i] between 0 and 255
var v = dataArray[i] / 255;
var y = v * height;
if(i === 0) {
canvasContext.moveTo(x, y);
} else {
canvasContext.lineTo(x, y);
}
x += sliceWidth;
}
canvasContext.lineTo(canvas.width, canvas.height/2);
// draw the path at once
canvasContext.stroke();
canvasContext.restore();
}
function clearCanvas() {
canvasContext.save();
// clear the canvas
// like this: canvasContext.clearRect(0, 0, width, height);
// Or use rgba fill to give a slight blur effect
canvasContext.fillStyle = 'rgba(0, 0, 0, 0.5)';
canvasContext.fillRect(0, 0, width, height);
canvasContext.restore();
}
function drawVolumeMeters() {
canvasContext.save();
// set the fill style to a nice gradient
canvasContext.fillStyle=gradient;
// left channel
analyserLeft.getByteFrequencyData(dataArrayLeft);
var averageLeft = getAverageVolume(dataArrayLeft);
// draw the vertical meter for left channel
canvasContext.fillRect(0,height-averageLeft,25,height);
// right channel
analyserRight.getByteFrequencyData(dataArrayRight);
var averageRight = getAverageVolume(dataArrayRight);
// draw the vertical meter for left channel
canvasContext.fillRect(26,height-averageRight,25,height);
canvasContext.restore();
}
function getAverageVolume(array) {
var values = 0;
var average;
var length = array.length;
// get all the frequency amplitudes
for (var i = 0; i < length; i++) {
values += array[i];
}
average = values / length;
return average;
}
function createStaticNodes() {
// these nodes can be used multiple times
stereoPanner = ctx.createStereoPanner();
// Create an analyser node
analyser = ctx.createAnalyser();
// Try changing for lower values: 512, 256, 128, 64...
analyser.fftSize = 1024;
bufferLength = analyser.frequencyBinCount;
dataArray = new Uint8Array(bufferLength);
// two analysers for the stereo volume meters
analyserLeft = ctx.createAnalyser();
analyserLeft.fftSize = 256;
bufferLengthLeft = analyserLeft.frequencyBinCount;
dataArrayLeft = new Uint8Array(bufferLengthLeft);
analyserRight = ctx.createAnalyser();
analyserRight.fftSize = 256;
bufferLengthRight = analyserRight.frequencyBinCount;
dataArrayRight = new Uint8Array(bufferLengthRight);
splitter = ctx.createChannelSplitter();
}
function makeSource(buffer) {
// build graph source -> gain -> compressor -> speakers
var source = ctx.createBufferSource();
var compressor = ctx.createDynamicsCompressor();
var gain = ctx.createGain();
gain.gain.value = 0.2 + Math.random();
source.buffer = buffer;
source.connect(gain);
gain.connect(compressor);
// connect the source node to a stereo pannel
compressor.connect(stereoPanner);
// connect the stereo pannel to the analyser
stereoPanner.connect(analyser);
// and teh analyser to the destination
analyser.connect(ctx.destination);
// This is new, we add another route from the stereoPanner node
// connect the source to the analyser and the splitter
stereoPanner.connect(splitter);
// connect one of the outputs from the splitter to
// the analyser
splitter.connect(analyserLeft,0,0);
splitter.connect(analyserRight,1,0);
return source;
}
function playSampleRepeated(buffer, rounds, interval, random, random2) {
if (typeof random == 'undefined') {
random = 0;
}
if (typeof random2 == 'undefined') {
random2 = 0;
}
var time = ctx.currentTime;
// Make multiple sources using the same buffer and play in quick succession.
for (var i = 0; i < rounds; i++) {
var source = makeSource(buffer);
source.playbackRate.value = 1 + Math.random() * random2;
source.start(time + i * interval + Math.random() * random);
}
}
function playSampleNormal(buffer){
var source = makeSource(buffer);
source.start();
}
function onSamplesDecoded(buffers){
console.log("all samples loaded and decoded");
shot1Normal.onclick = function(evt) {
playSampleNormal(buffers[0]);
};
shot2Normal.onclick = function(evt) {
playSampleNormal(buffers[1]);
};
shot1Repeated.onclick = function(evt) {
playSampleRepeated(buffers[0], 3, 0.1);
};
shot2Repeated.onclick = function(evt) {
playSampleRepeated(buffers[1], 10, 0.2);
};
shot1RepeatedIntervalRandom.onclick = function(evt) {
playSampleRepeated(buffers[0], 10, 0.08, 0.05);
};
shot2RepeatedPitchAndIntervalRandom.onclick = function(evt) {
playSampleRepeated(buffers[1], 10, 0.08, 1, 1);
};
}
function loadAllSoundSamples() {
bufferLoader = new BufferLoader(
ctx,
listOfSoundSamplesURLs, // urls of all sound samples to load
onSamplesDecoded // called when all samples have been loaded and decoded
);
bufferLoader.load(); // start loading. Will call onSamplesDecoded once all files loaded and decoded
}
/* ############################
BUFFER LOADER for loading multiple files asyncrhonously. The callback functions is called when all
files have been loaded and decoded
############################## */
function BufferLoader(context, urlList, callback) {
this.context = context;
this.urlList = urlList;
this.callback = callback;
this.bufferList = [];
this.loadCount = 0;
}
BufferLoader.prototype.loadBuffer = function(url, index) {
// Load buffer asynchronously
console.log('file : ' + url + "loading and decoding");
var request = new XMLHttpRequest();
request.open("GET", url, true);
request.responseType = "arraybuffer";
var loader = this;
request.onload = function() {
// Asynchronously decode the audio file data in request.response
loader.context.decodeAudioData(
request.response,
function(buffer) {
console.log("Loaded and decoded track " + (loader.loadCount+1) +
"/" + loader.urlList.length + "...");
if (!buffer) {
alert('error decoding file data: ' + url);
return;
}
loader.bufferList[index] = buffer;
//console.log("In bufferLoader.onload bufferList size is " + loader.bufferList.length + " index =" + index);
if (++loader.loadCount == loader.urlList.length)
// call the callback and pass it the decoded buffers, we've finihed
loader.callback(loader.bufferList);
},
function(error) {
console.error('decodeAudioData error', error);
}
);
};
request.onprogress = function(e) {
if(e.total !== 0) {
var percent = (e.loaded * 100) / e.total;
console.log("loaded " + percent + " % of song " + index);
}
};
request.onerror = function() {
alert('BufferLoader: XHR error');
};
request.send();
};
//noprotect
BufferLoader.prototype.load = function() {
console.log("Loading " + this.urlList.length + "track(s)... please wait...");
for (var i = 0; i < this.urlList.length; ++i)
this.loadBuffer(this.urlList[i], i);
};
Output
You can jump to the latest bin by adding /latest
to your URL
Keyboard Shortcuts
Shortcut | Action |
---|---|
ctrl + [num] | Toggle nth panel |
ctrl + 0 | Close focused panel |
ctrl + enter | Re-render output. If console visible: run JS in console |
Ctrl + l | Clear the console |
ctrl + / | Toggle comment on selected lines |
ctrl + ] | Indents selected lines |
ctrl + [ | Unindents selected lines |
tab | Code complete & Emmet expand |
ctrl + shift + L | Beautify code in active panel |
ctrl + s | Save & lock current Bin from further changes |
ctrl + shift + s | Open the share options |
ctrl + y | Archive Bin |
Complete list of JS Bin shortcuts |
JS Bin URLs
URL | Action |
---|---|
/ | Show the full rendered output. This content will update in real time as it's updated from the /edit url. |
/edit | Edit the current bin |
/watch | Follow a Code Casting session |
/embed | Create an embeddable version of the bin |
/latest | Load the very latest bin (/latest goes in place of the revision) |
/[username]/last | View the last edited bin for this user |
/[username]/last/edit | Edit the last edited bin for this user |
/[username]/last/watch | Follow the Code Casting session for the latest bin for this user |
/quiet | Remove analytics and edit button from rendered output |
.js | Load only the JavaScript for a bin |
.css | Load only the CSS for a bin |
Except for username prefixed urls, the url may start with http://jsbin.com/abc and the url fragments can be added to the url to view it differently. |