diff --git a/index.js b/index.js index 37d2343..4bf7417 100644 --- a/index.js +++ b/index.js @@ -161,12 +161,13 @@ aio.on('connection', function (socket) { device: 'auto_null.monitor', channels: 2, rate: 44100, - format: 'F32LE', + format: 'S16LE', }); record.on('connection', function(){ record.on('data', function(chunk) { - // Only send real audio data - if (chunk.length < 26456) { + // Only send non-zero audio data + let i16Array = Int16Array.from(chunk); + if (! i16Array.every(item => item === 0)) { aio.sockets.to(id).emit('audio', chunk); } }); diff --git a/package.json b/package.json index 3ef0aa2..c00ed48 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "kclient", - "version": "0.4.0", + "version": "0.4.1", "description": "Kclient is a wrapper for KasmVNC to add functionality to a containerized environment", "main": "index.js", "dependencies": { diff --git a/public/js/kclient.js b/public/js/kclient.js index bdd5aae..e932cc5 100644 --- a/public/js/kclient.js +++ b/public/js/kclient.js @@ -18,6 +18,75 @@ eventer(messageEvent,function(e) { } },false); +//// PCM player //// +var buffer = []; +var playing = false; +var lock = false; +// Check for audio stop to reset buffer +setInterval(function() { + if (playing) { + if (!lock) { + buffer = []; + playing = false; + } + lock = false; + } +}, 100); +function PCM() { + this.init() +} +// Player Init +PCM.prototype.init = function() { + // Establish audio context + this.audioCtx = new(window.AudioContext || window.webkitAudioContext)({ + sampleRate: 44100 + }) + this.audioCtx.resume() + this.gainNode = this.audioCtx.createGain() + this.gainNode.gain.value = 1 + this.gainNode.connect(this.audioCtx.destination) + this.startTime = this.audioCtx.currentTime +} +// Stereo player +PCM.prototype.feed = function(data) { + lock = true; + // Convert bytes to typed array then float32 array + let i16Array = new Int16Array(data, 0, data.length); + let f32Array = Float32Array.from(i16Array, x => x / 32767); + buffer = new Float32Array([...buffer, ...f32Array]); + let buffAudio = this.audioCtx.createBuffer(2, buffer.length, 44100); + let duration = buffAudio.duration / 2; + if ((duration > .05) || (playing)) { + playing = true; + let buffSource = this.audioCtx.createBufferSource(); + let arrLength = buffer.length / 2; + let left = buffAudio.getChannelData(0); + let right = buffAudio.getChannelData(1); + let byteCount = 0; + let offset = 1; + for (let count = 0; count < arrLength; count++) { + left[count] = buffer[byteCount]; + byteCount += 2; + right[count] = buffer[offset]; + offset += 2; + } + buffer = []; + if (this.startTime < this.audioCtx.currentTime) { + this.startTime = this.audioCtx.currentTime; + } + buffSource.buffer = buffAudio; + buffSource.connect(this.gainNode); + buffSource.start(this.startTime); + this.startTime += duration; + } +} +// Destroy player +PCM.prototype.destroy = function() { + buffer = []; + playing = false; + this.audioCtx.close(); + this.audioCtx = null; +}; // Handle Toggle divs function openToggle(id) { @@ -76,7 +145,7 @@ function audio() { return; } socket.emit('open', ''); - player = new PCMPlayer(); + player = new PCM(); $('#audioButton').addClass("icons-selected"); } diff --git a/public/js/pcm-player.js b/public/js/pcm-player.js deleted file mode 100644 index 0014f10..0000000 --- a/public/js/pcm-player.js +++ /dev/null @@ -1 +0,0 @@ -function PCMPlayer(t){this.init(t)}PCMPlayer.prototype.init=function(){this.audioCtx=new(window.AudioContext||window.webkitAudioContext)({sampleRate:44100}),this.audioCtx.resume(),this.gainNode=this.audioCtx.createGain(),this.gainNode.gain.value=1,this.gainNode.connect(this.audioCtx.destination),this.startTime=this.audioCtx.currentTime},PCMPlayer.prototype.feed=function(t){t=new Float32Array(t);let i=this.audioCtx.createBufferSource(),e=t.length/2,a=this.audioCtx.createBuffer(2,e,44100),o=a.getChannelData(0),n=a.getChannelData(1),s=0;r=1;for(let u=0;u", "short_name": "<%- title -%>", "manifest_version": 2, - "version": "0.4.0", + "version": "0.4.1", "display": "fullscreen", "background_color": "#000000", "theme_color": "#000000",