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