<div id=time>Time domain data
<canvas id=scope></canvas>
</div>
<div id=frequency>
Frequency domain data
<canvas id=freq></canvas>
</div>
<pre>
// gain
var g = ctx.createGain();
g.connect(sink);
source.connect(g);
g.gain.setValueAtTime(0, 0);
g.gain.linearRampToValueAtTime(1.0, 5);
// delay
var d = ctx.createDelay();
var g = ctx.createGain();
d.connect(g);
g.gain.value = 0.5;
g.connect(sink);
source.connect(d);
source.connect(sink);
g.connect(d);
d.delayTime.value = 0.1;
source.connect(d);
// biquad
var b = ctx.createBiquadFilter();
b.frequency.value = 1500;
b.connect(sink);
source.connect(b);
// Waveshaper
function makeDistortionCurve( amount ) {
var k = typeof amount === 'number' ? amount : 50,
n_samples = 44100,
curve = new Float32Array(n_samples),
deg = Math.PI / 180,
i = 0,
x;
for ( ; i < n_samples; ++i ) {
x = i * 2 / n_samples - 1;
curve[i] = ( 3 + k ) * x * 20 * deg / ( Math.PI + k * Math.abs(x) );
}
return curve;
}
var ws = ctx.createWaveShaper();
ws.curve = makeDistortionCurve();
source.connect(ws);
ws.connect(sink);
// verb
getFile(url, function(data) {
getFile(verb, function(verb) {
var source = ctx.createBufferSource();
source.buffer = data;
source.loop = true;
source.playbackRate.value = 1.44;
var c = ctx.createConvolver();
c.buffer = verb;
source.connect(c);
c.connect(sink);
source.start(0);
});
});
</pre>
body {
background-color: #269;
background-size: 100px 100px, 100px 100px, 20px 20px, 20px 20px;
background-position: -2px -2px, -2px -2px, -1px -1px, -1px -1px;
background-image: linear-gradient(white 2px, transparent 2px),
linear-gradient(0, white 2px, transparent 2px),
linear-gradient(rgba(255,255,255,.3) 1px, transparent 1px),
linear-gradient(0, rgba(255,255,255,.3) 1px, transparent 1px);
background-image: linear-gradient(white 2px, transparent 2px),
linear-gradient(0, white 2px, transparent 2px),
linear-gradient(rgba(255,255,255,.3) 1px, transparent 1px),
linear-gradient(0, rgba(255,255,255,.3) 1px, transparent 1px);
background-image: linear-gradient(rgba(255,255,255,.2) 2px, transparent 2px),
linear-gradient(90deg, rgba(255,255,255,.2) 2px, transparent 2px),
linear-gradient(rgba(255,255,255,.2) 1px, transparent 1px),
linear-gradient(90deg, rgba(255,255,255,.2) 1px, transparent 1px);
color: white;
font-family: courier;
}
div, pre {
border: 1px solid white;
padding: 1em;
background-color: rgba(255, 255, 255, 0.2);
margin-bottom: 1em;
}
canvas {
display: block;
}
var cvs = document.getElementById("scope");
var cvswidth = 512;
cvs.width = cvswidth;
cvs.height = 256;
var c = cvs.getContext("2d");
c.fillStyle = "#fff";
var cvs2 = document.getElementById("freq");
cvs2.width = cvswidth;
cvs2.height = 256;
var c2 = cvs2.getContext("2d");
c2.fillStyle = "#fff";
// Arrays for the analysis
var array = new Uint8Array(cvswidth);
var fft = new Float32Array(cvswidth);
// render loop
function render() {
c.clearRect(0, 0, cvs.width, cvs.height);
c2.clearRect(0, 0, cvs.width, cvs.height);
sink.getByteTimeDomainData(array);
sink.getFloatFrequencyData(fft);
var minDecibels = -sink.minDecibels;
var range = -sink.minDecibels + sink.maxDecibels;
for (var i = 0; i < array.length; i++) {
// data is unitless 0-255
c.fillRect(i * 2, 128, 0.8, array[i] - 127);
// normalize to the canvas' size, data goes from -100 to -30 by default
c2.fillRect(i * 2, cvs2.height, 0.8, -((fft[i] + minDecibels) / range) * cvs2.height);
}
requestAnimationFrame(render);
}
// Get a context
var ctx = new AudioContext();
// Get an analyser
var sink = ctx.createAnalyser();
// Set the size of the fft analysis
sink.frequencyBinCount = 512;
// Connect it to the destination
sink.connect(ctx.destination);
// Handy cross-origin file
// Handy cross-origin file
var url = "http://corsify.appspot.com/http://paul.cx/public/think-looped-mono.wav";
var verb = "http://corsify.appspot.com/http://paul.cx/public/verb.wav";
// Function to get a file from a URL, decode it,
function getFile(url, cb) {
var request = new XMLHttpRequest();
request.open("GET", url, true);
request.responseType = "arraybuffer";
request.onload = function() {
ctx.decodeAudioData(request.response, function(data) {
cb(data);
}, function() {
alert("decodeAudioData error");
});
};
request.send();
}
/**** Your code here ****/
getFile(url, function(data) {
getFile(verb, function(verb) {
var source = ctx.createBufferSource();
source.buffer = data;
source.loop = true;
source.playbackRate.value = 1.44;
source.connect(sink);
source.start(0);
});
});
/**** ****/
render();
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. |