rlm@41: package com.aurellem.capture.examples; rlm@41: rlm@41: import java.io.File; rlm@41: import java.io.IOException; rlm@42: import java.lang.reflect.Field; rlm@41: import java.nio.ByteBuffer; rlm@41: rlm@41: import javax.sound.sampled.AudioFormat; rlm@41: rlm@41: import org.tritonus.share.sampled.FloatSampleTools; rlm@41: rlm@45: import com.aurellem.capture.AurellemSystemDelegate; rlm@41: import com.aurellem.capture.Capture; rlm@41: import com.aurellem.capture.IsoTimer; rlm@41: import com.aurellem.capture.audio.CompositeSoundProcessor; rlm@41: import com.aurellem.capture.audio.MultiListener; rlm@41: import com.aurellem.capture.audio.SoundProcessor; rlm@41: import com.aurellem.capture.audio.WaveFileWriter; rlm@41: import com.jme3.app.SimpleApplication; rlm@41: import com.jme3.audio.AudioNode; rlm@41: import com.jme3.audio.Listener; rlm@41: import com.jme3.cinematic.MotionPath; rlm@42: import com.jme3.cinematic.events.AbstractCinematicEvent; rlm@41: import com.jme3.cinematic.events.MotionTrack; rlm@41: import com.jme3.material.Material; rlm@41: import com.jme3.math.ColorRGBA; rlm@41: import com.jme3.math.FastMath; rlm@41: import com.jme3.math.Quaternion; rlm@41: import com.jme3.math.Vector3f; rlm@41: import com.jme3.scene.Geometry; rlm@41: import com.jme3.scene.Node; rlm@41: import com.jme3.scene.shape.Box; rlm@41: import com.jme3.scene.shape.Sphere; rlm@41: import com.jme3.system.AppSettings; rlm@45: import com.jme3.system.JmeSystem; rlm@41: rlm@41: /** rlm@41: * rlm@56: * Demonstrates advanced use of the audio capture and recording rlm@56: * features. Multiple perspectives of the same scene are rlm@56: * simultaneously rendered to different sound files. rlm@41: * rlm@56: * A key limitation of the way multiple listeners are implemented is rlm@56: * that only 3D positioning effects are realized for listeners other rlm@56: * than the main LWJGL listener. This means that audio effects such rlm@56: * as environment settings will *not* be heard on any auxiliary rlm@56: * listeners, though sound attenuation will work correctly. rlm@41: * rlm@56: * Multiple listeners as realized here might be used to make AI rlm@56: * entities that can each hear the world from their own perspective. rlm@41: * rlm@41: * @author Robert McIntyre rlm@41: */ rlm@41: rlm@41: public class Advanced extends SimpleApplication { rlm@41: rlm@62: /** rlm@62: * You will see three grey cubes, a blue sphere, and a path which rlm@62: * circles each cube. The blue sphere is generating a constant rlm@62: * monotone sound as it moves along the track. Each cube is rlm@62: * listening for sound; when a cube hears sound whose intensity is rlm@62: * greater than a certain threshold, it changes its color from rlm@62: * grey to green. rlm@62: * rlm@62: * Each cube is also saving whatever it hears to a file. The rlm@62: * scene from the perspective of the viewer is also saved to a rlm@62: * video file. When you listen to each of the sound files rlm@62: * alongside the video, the sound will get louder when the sphere rlm@62: * approaches the cube that generated that sound file. This rlm@62: * shows that each listener is hearing the world from its own rlm@62: * perspective. rlm@62: * rlm@62: */ rlm@62: public static void main(String[] args) { rlm@62: Advanced app = new Advanced(); rlm@62: AppSettings settings = new AppSettings(true); rlm@62: settings.setAudioRenderer(AurellemSystemDelegate.SEND); rlm@62: JmeSystem.setSystemDelegate(new AurellemSystemDelegate()); rlm@62: app.setSettings(settings); rlm@62: app.setShowSettings(false); rlm@62: app.setPauseOnLostFocus(false); rlm@43: rlm@62: try { rlm@62: //Capture.captureVideo(app, File.createTempFile("advanced",".avi")); rlm@62: Capture.captureAudio(app, File.createTempFile("advanced", ".wav")); rlm@62: } rlm@62: catch (IOException e) {e.printStackTrace();} rlm@62: rlm@62: app.start(); rlm@43: } rlm@41: rlm@56: rlm@62: private Geometry bell; rlm@62: private Geometry ear1; rlm@62: private Geometry ear2; rlm@62: private Geometry ear3; rlm@62: private AudioNode music; rlm@62: private MotionTrack motionControl; rlm@62: private IsoTimer motionTimer = new IsoTimer(60); rlm@56: rlm@62: private Geometry makeEar(Node root, Vector3f position){ rlm@62: Material mat = new Material(assetManager, "Common/MatDefs/Misc/Unshaded.j3md"); rlm@62: Geometry ear = new Geometry("ear", new Box(1.0f, 1.0f, 1.0f)); rlm@62: ear.setLocalTranslation(position); rlm@62: mat.setColor("Color", ColorRGBA.Green); rlm@62: ear.setMaterial(mat); rlm@62: root.attachChild(ear); rlm@62: return ear; rlm@62: } rlm@56: rlm@62: private Vector3f[] path = new Vector3f[]{ rlm@62: // loop 1 rlm@62: new Vector3f(0, 0, 0), rlm@62: new Vector3f(0, 0, -10), rlm@62: new Vector3f(-2, 0, -14), rlm@62: new Vector3f(-6, 0, -20), rlm@62: new Vector3f(0, 0, -26), rlm@62: new Vector3f(6, 0, -20), rlm@62: new Vector3f(0, 0, -14), rlm@62: new Vector3f(-6, 0, -20), rlm@62: new Vector3f(0, 0, -26), rlm@62: new Vector3f(6, 0, -20), rlm@62: // loop 2 rlm@62: new Vector3f(5, 0, -5), rlm@62: new Vector3f(7, 0, 1.5f), rlm@62: new Vector3f(14, 0, 2), rlm@62: new Vector3f(20, 0, 6), rlm@62: new Vector3f(26, 0, 0), rlm@62: new Vector3f(20, 0, -6), rlm@62: new Vector3f(14, 0, 0), rlm@62: new Vector3f(20, 0, 6), rlm@62: new Vector3f(26, 0, 0), rlm@62: new Vector3f(20, 0, -6), rlm@62: new Vector3f(14, 0, 0), rlm@62: // loop 3 rlm@62: new Vector3f(8, 0, 7.5f), rlm@62: new Vector3f(7, 0, 10.5f), rlm@62: new Vector3f(6, 0, 20), rlm@62: new Vector3f(0, 0, 26), rlm@62: new Vector3f(-6, 0, 20), rlm@62: new Vector3f(0, 0, 14), rlm@62: new Vector3f(6, 0, 20), rlm@62: new Vector3f(0, 0, 26), rlm@62: new Vector3f(-6, 0, 20), rlm@62: new Vector3f(0, 0, 14), rlm@62: // begin ellipse rlm@62: new Vector3f(16, 5, 20), rlm@62: new Vector3f(0, 0, 26), rlm@62: new Vector3f(-16, -10, 20), rlm@62: new Vector3f(0, 0, 14), rlm@62: new Vector3f(16, 20, 20), rlm@62: new Vector3f(0, 0, 26), rlm@62: new Vector3f(-10, -25, 10), rlm@62: new Vector3f(-10, 0, 0), rlm@62: // come at me! rlm@62: new Vector3f(-28.00242f, 48.005623f, -34.648228f), rlm@62: new Vector3f(0, 0 , -20), rlm@62: }; rlm@56: rlm@62: private void createScene() { rlm@62: Material mat = new Material(assetManager, "Common/MatDefs/Misc/Unshaded.j3md"); rlm@62: bell = new Geometry( "sound-emitter" , new Sphere(15,15,1)); rlm@62: mat.setColor("Color", ColorRGBA.Blue); rlm@62: bell.setMaterial(mat); rlm@62: rootNode.attachChild(bell); rlm@62: rlm@62: ear1 = makeEar(rootNode, new Vector3f(0, 0 ,-20)); rlm@62: ear2 = makeEar(rootNode, new Vector3f(0, 0 ,20)); rlm@62: ear3 = makeEar(rootNode, new Vector3f(20, 0 ,0)); rlm@62: rlm@62: MotionPath track = new MotionPath(); rlm@62: rlm@62: for (Vector3f v : path){ rlm@62: track.addWayPoint(v); rlm@62: } rlm@62: track.setCurveTension(0.80f); rlm@62: rlm@62: motionControl = new MotionTrack(bell,track); rlm@62: // for now, use reflection to change the timer... rlm@62: // motionControl.setTimer(new IsoTimer(60)); rlm@62: rlm@62: try { rlm@62: Field timerField; rlm@62: timerField = AbstractCinematicEvent.class.getDeclaredField("timer"); rlm@62: timerField.setAccessible(true); rlm@62: try {timerField.set(motionControl, motionTimer);} rlm@62: catch (IllegalArgumentException e) {e.printStackTrace();} rlm@62: catch (IllegalAccessException e) {e.printStackTrace();} rlm@62: } rlm@62: catch (SecurityException e) {e.printStackTrace();} rlm@62: catch (NoSuchFieldException e) {e.printStackTrace();} rlm@62: rlm@62: rlm@62: motionControl.setDirectionType(MotionTrack.Direction.PathAndRotation); rlm@62: motionControl.setRotation(new Quaternion().fromAngleNormalAxis(-FastMath.HALF_PI, Vector3f.UNIT_Y)); rlm@62: motionControl.setInitialDuration(20f); rlm@62: motionControl.setSpeed(1f); rlm@62: rlm@62: track.enableDebugShape(assetManager, rootNode); rlm@62: positionCamera(); rlm@56: } rlm@56: rlm@56: rlm@62: private void positionCamera(){ rlm@62: this.cam.setLocation(new Vector3f(-28.00242f, 48.005623f, -34.648228f)); rlm@62: this.cam.setRotation(new Quaternion(0.3359635f, 0.34280345f, -0.13281013f, 0.8671653f)); rlm@62: } rlm@56: rlm@62: private void initAudio() { rlm@62: org.lwjgl.input.Mouse.setGrabbed(false); rlm@62: music = new AudioNode(assetManager, "Sound/Effects/Beep.ogg", false); rlm@56: rlm@62: rootNode.attachChild(music); rlm@62: audioRenderer.playSource(music); rlm@62: music.setPositional(true); rlm@62: music.setVolume(1f); rlm@62: music.setReverbEnabled(false); rlm@62: music.setDirectional(false); rlm@62: music.setMaxDistance(200.0f); rlm@62: music.setRefDistance(1f); rlm@62: //music.setRolloffFactor(1f); rlm@62: music.setLooping(false); rlm@62: audioRenderer.pauseSource(music); rlm@62: } rlm@56: rlm@62: public class Dancer implements SoundProcessor { rlm@62: Geometry entity; rlm@62: float scale = 2; rlm@62: public Dancer(Geometry entity){ rlm@62: this.entity = entity; rlm@62: } rlm@56: rlm@62: /** rlm@62: * this method is irrelevant since there is no state to cleanup. rlm@62: */ rlm@62: public void cleanup() {} rlm@56: rlm@41: rlm@62: /** rlm@62: * Respond to sound! This is the brain of an AI entity that rlm@62: * hears its surroundings and reacts to them. rlm@62: */ rlm@62: public void process(ByteBuffer audioSamples, int numSamples, AudioFormat format) { rlm@62: audioSamples.clear(); rlm@62: byte[] data = new byte[numSamples]; rlm@62: float[] out = new float[numSamples]; rlm@62: audioSamples.get(data); rlm@62: FloatSampleTools.byte2floatInterleaved(data, 0, out, 0, rlm@62: numSamples/format.getFrameSize(), format); rlm@41: rlm@62: float max = Float.NEGATIVE_INFINITY; rlm@62: for (float f : out){if (f > max) max = f;} rlm@62: audioSamples.clear(); rlm@41: rlm@62: if (max > 0.1){entity.getMaterial().setColor("Color", ColorRGBA.Green);} rlm@62: else {entity.getMaterial().setColor("Color", ColorRGBA.Gray);} rlm@62: } rlm@41: } rlm@62: rlm@62: private void prepareEar(Geometry ear, int n){ rlm@62: if (this.audioRenderer instanceof MultiListener){ rlm@62: MultiListener rf = (MultiListener)this.audioRenderer; rlm@62: rlm@62: Listener auxListener = new Listener(); rlm@62: auxListener.setLocation(ear.getLocalTranslation()); rlm@62: rlm@62: rf.addListener(auxListener); rlm@62: WaveFileWriter aux = null; rlm@62: rlm@62: try {aux = new WaveFileWriter(File.createTempFile("advanced-audio-" + n, ".wav"));} rlm@62: catch (IOException e) {e.printStackTrace();} rlm@62: rlm@62: rf.registerSoundProcessor(auxListener, rlm@62: new CompositeSoundProcessor(new Dancer(ear), aux)); rlm@62: rlm@62: } rlm@62: } rlm@62: rlm@62: rlm@62: public void simpleInitApp() { rlm@62: this.setTimer(new IsoTimer(60)); rlm@62: initAudio(); rlm@62: rlm@62: createScene(); rlm@62: rlm@62: prepareEar(ear1, 1); rlm@62: prepareEar(ear2, 1); rlm@62: prepareEar(ear3, 1); rlm@62: rlm@62: motionControl.play(); rlm@62: rlm@62: } rlm@62: rlm@62: public void simpleUpdate(float tpf) { rlm@62: motionTimer.update(); rlm@62: if (music.getStatus() != AudioNode.Status.Playing){ rlm@62: music.play(); rlm@62: } rlm@62: Vector3f loc = cam.getLocation(); rlm@62: Quaternion rot = cam.getRotation(); rlm@62: listener.setLocation(loc); rlm@62: listener.setRotation(rot); rlm@62: music.setLocalTranslation(bell.getLocalTranslation()); rlm@62: } rlm@41: rlm@41: }