com.badlogic.gdx.backends.gwt.webaudio.WebAudioAPIManager Maven / Gradle / Ivy
/*******************************************************************************
* Copyright 2011 See AUTHORS file.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
******************************************************************************/
package com.badlogic.gdx.backends.gwt.webaudio;
import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.LifecycleListener;
import com.badlogic.gdx.audio.Music;
import com.badlogic.gdx.audio.Sound;
import com.badlogic.gdx.backends.gwt.GwtFileHandle;
import com.badlogic.gdx.backends.gwt.preloader.AssetDownloader;
import com.badlogic.gdx.files.FileHandle;
import com.google.gwt.core.client.JavaScriptObject;
import com.google.gwt.media.client.Audio;
import com.google.gwt.typedarrays.shared.ArrayBuffer;
import com.google.gwt.typedarrays.shared.Int8Array;
import com.google.gwt.typedarrays.shared.TypedArrays;
import com.google.gwt.xhr.client.ReadyStateChangeHandler;
import com.google.gwt.xhr.client.XMLHttpRequest;
import com.google.gwt.xhr.client.XMLHttpRequest.ResponseType;
public class WebAudioAPIManager implements LifecycleListener {
private final JavaScriptObject audioContext;
private final JavaScriptObject globalVolumeNode;
private final AssetDownloader assetDownloader;
private final AudioControlGraphPool audioControlGraphPool;
private static boolean soundUnlocked;
public WebAudioAPIManager () {
this.assetDownloader = new AssetDownloader();
this.audioContext = createAudioContextJSNI();
this.globalVolumeNode = createGlobalVolumeNodeJSNI();
this.audioControlGraphPool = new AudioControlGraphPool(audioContext, globalVolumeNode);
// for automatically muting/unmuting on pause/resume
Gdx.app.addLifecycleListener(this);
/*
* The Web Audio API is blocked on many platforms until the developer triggers the first sound playback using the API. But
* it MUST happen as a direct result of a few specific input events. This is a major point of confusion for developers new
* to the platform. Here we attach event listeners to the graphics canvas in order to unlock the sound system on the first
* input event. On the event, we play a silent sample, which should unlock the sound - on platforms where it is not
* necessary the effect should not be noticeable (i.e. we play silence). As soon as the attempt to unlock has been
* performed, we remove all the event listeners.
*/
if (isAudioContextLocked(audioContext))
hookUpSoundUnlockers();
else
setUnlocked();
}
public native void hookUpSoundUnlockers () /*-{
var self = this;
var audioContext = [email protected]::audioContext;
// An array of various user interaction events we should listen for
var userInputEventNames = [
'click', 'contextmenu', 'auxclick', 'dblclick', 'mousedown',
'mouseup', 'pointerup', 'touchend', 'keydown', 'keyup', 'touchstart'
];
var unlock = function(e) {
// resume audio context if it was suspended. It's only required for musics since sounds automatically resume
// audio context when started.
audioContext.resume();
[email protected]::setUnlocked()();
userInputEventNames.forEach(function (eventName) {
$doc.removeEventListener(eventName, unlock);
});
};
userInputEventNames.forEach(function (eventName) {
$doc.addEventListener(eventName, unlock);
});
}-*/;
public void setUnlocked () {
Gdx.app.log("Webaudio", "Audiocontext unlocked");
soundUnlocked = true;
}
public void setSinkId (String sinkId) {
setSinkIdJSNI(audioContext, sinkId);
}
public static boolean isSoundUnlocked () {
return soundUnlocked;
}
static native boolean isAudioContextLocked (JavaScriptObject audioContext) /*-{
return audioContext.state !== 'running';
}-*/;
/** Older browsers do not support the Web Audio API. This is where we find out.
*
* @return is the WebAudioAPI available in this browser? */
public static native boolean isSupported () /*-{
return typeof (window.AudioContext || window.webkitAudioContext) != "undefined";
}-*/;
private static native JavaScriptObject createAudioContextJSNI () /*-{
var AudioContext = window.AudioContext || window.webkitAudioContext;
if (AudioContext) {
var audioContext = new AudioContext();
return audioContext;
}
return null;
}-*/;
private static native void setSinkIdJSNI (JavaScriptObject audioContext, String sinkId) /*-{
if (!audioContext.setSinkId) return;
audioContext.setSinkId(sinkId);
}-*/;
private native JavaScriptObject createGlobalVolumeNodeJSNI () /*-{
var audioContext = [email protected]::audioContext;
var gainNode = null;
if (audioContext.createGain)
// Standard compliant
gainNode = audioContext.createGain();
else
// Old WebKit/iOS
gainNode = audioContext.createGainNode();
// Default to full, unmuted volume
gainNode.gain.value = 1.0;
// Connect the global volume to the speakers. This will be the last part of our audio graph.
gainNode.connect(audioContext.destination);
return gainNode;
}-*/;
private native void disconnectJSNI () /*-{
var audioContext = [email protected]::audioContext;
var gainNode = [email protected]::globalVolumeNode;
gainNode.disconnect(audioContext.destination);
}-*/;
private native void connectJSNI () /*-{
var audioContext = [email protected]::audioContext;
var gainNode = [email protected]::globalVolumeNode;
gainNode.connect(audioContext.destination);
}-*/;
public JavaScriptObject getAudioContext () {
return audioContext;
}
public Sound createSound (FileHandle fileHandle) {
final WebAudioAPISound newSound = new WebAudioAPISound(audioContext, globalVolumeNode, audioControlGraphPool);
String url = ((GwtFileHandle)fileHandle).getAssetUrl();
XMLHttpRequest request = XMLHttpRequest.create();
request.setOnReadyStateChange(new ReadyStateChangeHandler() {
@Override
public void onReadyStateChange (XMLHttpRequest xhr) {
if (xhr.getReadyState() == XMLHttpRequest.DONE) {
if (xhr.getStatus() != 200) {
} else {
Int8Array data = TypedArrays.createInt8Array(xhr.getResponseArrayBuffer());
/*
* Start decoding the sound data. This is an asynchronous process, which is a bad fit for the libGDX API, which
* expects sound creation to be synchronous. The result is that sound won't actually start playing until the
* decoding is done.
*/
decodeAudioData(getAudioContext(), data.buffer(), newSound);
}
}
}
});
request.open("GET", url);
request.setResponseType(ResponseType.ArrayBuffer);
request.send();
return newSound;
}
public Music createMusic (FileHandle fileHandle) {
String url = ((GwtFileHandle)fileHandle).getAssetUrl();
Audio audio = Audio.createIfSupported();
audio.setSrc(url);
WebAudioAPIMusic music = new WebAudioAPIMusic(audioContext, audio, audioControlGraphPool);
return music;
}
public static native void decodeAudioData (JavaScriptObject audioContextIn, ArrayBuffer audioData,
WebAudioAPISound targetSound) /*-{
audioContextIn
.decodeAudioData(
audioData,
function(buffer) {
targetSound.@com.badlogic.gdx.backends.gwt.webaudio.WebAudioAPISound::setAudioBuffer(Lcom/google/gwt/core/client/JavaScriptObject;)(buffer);
}, function() {
console.log("Error: decodeAudioData");
});
}-*/;
@Override
public void pause () {
// As the web application looses focus, we mute the sound
disconnectJSNI();
}
@Override
public void resume () {
// As the web application regains focus, we unmute the sound
connectJSNI();
}
public void setGlobalVolume (float volume) {
setGlobalVolumeJSNI(volume);
}
public native JavaScriptObject setGlobalVolumeJSNI (float volume) /*-{
var gainNode = [email protected]::globalVolumeNode;
gainNode.gain.value = volume;
}-*/;
@Override
public void dispose () {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy