275 lines
6.5 KiB
JavaScript

let analyser;
let context;
let source;
let fftsize = 2048;
let oscillator = true;
let log;
let percent;
let times;
let speed = 0.1;
let bufferLength;
let labels;
let color = true;
let width = window.innerwidth;
let height = window.innerheight;
let canvas = document.getElementById('spectrogram-canvas');
let ctx = canvas.getContext('2d');
let lastRenderTime = new Date();
let instantaneousFPS = 0;
let audio = document.getElementById('audio');
document.getElementById("launch-larynx").addEventListener('click', function() {
loadSound(audio.src)
startLarynx();
});
function loadSound(url) {
let AudioContext = window.AudioContext || window.webkitAudioContext || window.mozAudioContext;
context = new AudioContext();
request = new XMLHttpRequest();
request.open('GET', audio.src, true);
request.responseType = 'arraybuffer';
request.addEventListener('load', bufferSound, false);
request.send();
}
function bufferSound(event) {
request = event.target;
isource = context.createBufferSource();
isource.buffer = context.createBuffer(request.response, false);
source = isource;
}
function startLarynx() {
analyser = context.createAnalyser();
render();
}
function render() {
let didResize = false;
analyser = context.createAnalyser();
// Ensure dimensions are accurate.
if (canvas.width != width) {
canvas.width = width;
didResize = true;
}
if (canvas.height != height) {
canvas.height = height;
didResize = true;
}
renderTimeDomain();
renderFreqDomain();
if (labels && didResize) {
renderAxesLabels();
}
let now = new Date();
if (lastRenderTime) {
instantaneousFPS = now - lastRenderTime;
}
lastRenderTime = now;
}
function draw() {
drawVisual = requestAnimationFrame(draw);
analyser.getByteTimeDomainData(times);
ctx.fillStyle = 'rgb(200, 200, 200)';
ctx.fillRect(0, 0, width, height);
ctx.linewidth = 2;
ctx.strokeStyle = 'rgb(0, 0, 0)';
ctx.beginPath();
var slicewidth = width * 1.0 / bufferLength;
var x = 0;
for (var i = 0; i < bufferLength; i++) {
var v = times[i] / 128.0;
var y = v * height / 2;
if (i === 0) {
ctx.moveTo(x, y);
} else {
ctx.lineTo(x, y);
}
x += slicewidth;
}
ctx.lineTo(canvas.width, canvas.height / 2);
ctx.stroke();
}
function renderTimeDomain() {
bufferLength = analyser.frequencyBinCount;
times = new Uint8Array(bufferLength);
draw();
}
function renderFreqDomain() {
let freq = new Uint8Array(analyser.frequencyBinCount);
analyser.getByteFrequencyData(freq);
for (let i = 0; i < freq.length; i++) {
let value;
// Draw each pixel with the specific color.
if (log) {
logIndex = logScale(i, freq.length);
value = freq[logIndex];
} else {
value = freq[i];
}
ctx.fillStyle = (color ? getFullColor(value) : getGrayColor(value));
let perceint = i / freq.length;
let y = Math.round(percent * height);
// Draw the line at the right side of the canvas.
ctx.fillRect(width - speed, height - y, speed, speed);
}
ctx.translate(-speed, 0);
// Reset the transformation matrix
ctx.setTransform(1, 0, 0, 1, 0, 0);
}
function logScale(index, total, opt_base) {
let base = opt_base || 2;
let logmax = logBase(total + 1, base);
let exp = logmax * index / total;
return Math.round(Math.pow(base, exp) - 1);
}
function logBase(val, base) {
return Math.log(val) / Math.log(base);
}
function renderAxesLabels() {
canvas.width = width;
canvas.height = height;
let ctx = canvas.getContext('2d');
let startFreq = 440;
let nyquist = context.sampleRate / 2;
let endFreq = nyquist - startFreq;
let step = (endFreq - startFreq) / ticks;
let yLabelOffset = 5;
// Render the vertical frequency axis.
for (let i = 0; i <= ticks; i++) {
let freq = startFreq + (step * i);
// Get the y coordinate from the current label.
let index = freqToIndex(freq);
let percent = index / getFFTBinCount();
let y = (1 - percent) * height;
let x = width - 60;
// Get the value for the current y coordinate.
let label;
if (log) {
// Handle a logarithmic scale.
let logIndex = logScale(index, getFFTBinCount());
// Never show 0 Hz.
freq = Math.max(1, indexToFreq(logIndex));
}
label = formatFreq(freq);
let units = formatUnits(freq);
ctx.font = '16px Inconsolata';
// Draw the value.
ctx.textAlign = 'right';
ctx.fillText(label, x, y + yLabelOffset);
// Draw the units.
ctx.textAlign = 'left';
ctx.fillText(units, x + 10, y + yLabelOffset);
// Draw a tick mark.
ctx.fillRect(x + 40, y, 30, 2);
}
}
function clearAxesLabels() {
let canvas = labels;
let ctx = canvas.getContext('2d');
ctx.clearRect(0, 0, width, height);
}
function formatFreq(freq) {
return (freq >= 1000 ? (freq / 1000).toFixed(1) : Math.round(freq));
}
function formatUnits(freq) {
return (freq >= 1000 ? 'KHz' : 'Hz');
}
function indexToFreq(index) {
let nyquist = context.sampleRate / 2;
return nyquist / getFFTBinCount() * index;
}
function freqToIndex(frequency) {
let nyquist = context.sampleRate / 2;
return Math.round(frequency / nyquist * getFFTBinCount());
}
function getFFTBinCount() {
return fftsize / 2;
}
// function onStream(stream) {
// let input = context.createMediaStreamSource(stream);
// let analyser = context.createAnalyser();
// analyser.smoothingTimeConstant = 0;
// analyser.fftSize = fftsize;
// // Connect graph.
// input.connect(analyser);
// // Setup a timer to visualize some stuff.
// render();
// }
function onStreamError(e) {
console.error(e);
}
function getGrayColor(value) {
return 'rgb(V, V, V)'.replace(/V/g, 255 - value);
}
function getFullColor(value) {
let fromH = 62;
let toH = 0;
let percent = value / 255;
let delta = percent * (toH - fromH);
let hue = fromH + delta;
return 'hsl(H, 100%, 50%)'.replace(/H/g, hue);
}
function logChanged() {
if (labels) {
renderAxesLabels();
}
}
function ticksChanged() {
if (labels) {
renderAxesLabels();
}
}
function labelsChanged() {
if (labels) {
renderAxesLabels();
} else {
clearAxesLabels();
}
}