[译文]JOAL教程
原文地址:http://jogamp.org/joal-demos/www/devmaster/lesson2.html
原文作者:Athomas Goldberg
译文:三向板砖
转载请保留以上信息。
本节对应的连续代码页及学习笔记:http://blog.csdn.net/shuzhe66/article/details/40260465
第二课 循环与淡出
本文是DevMaster.net(http://devmaster.net/)的OpenAL教程对应的JOAL版本。C语言版原文作者为JesseMaurais
希望上一课对你来说有点用,本次将会是一个简单、快速的教程,当然它也难不到哪去。
import java.nio.ByteBuffer; import com.jogamp.openal.AL; import com.jogamp.openal.ALC; import com.jogamp.openal.ALFactory; import com.jogamp.openal.util.ALut; public class LoopingAndFadeaway { static int[] buffer = new int[1]; static int[] source = new int[1]; static float[] sourcePos = { 0.0f, 0.0f, 0.0f }; static float[] sourceVel = { 0.0f, 0.0f, 0.1f }; static float[] listenerPos = { 0.0f, 0.0f, 0.0f }; static float[] listenerVel = { 0.0f, 0.0f, 0.0f }; static float[] listenerOri = { 0.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f }; static AL al; static ALC alc;
此处唯一一个与上节不同的地方是声源的速度,我们给了z轴一个0.1的速度。
static int loadALData() { if (al.alGetError() != AL.AL_NO_ERROR) { return AL.AL_FALSE; } int[] format = new int[1]; int[] size = new int[1]; ByteBuffer[] data = new ByteBuffer[1]; int[] freq = new int[1]; int[] loop = new int[1]; //将wav数据装入缓冲区 al.alGenBuffers(1, buffer, 0); if (al.alGetError() != AL.AL_NO_ERROR) return AL.AL_FALSE; ALut.alutLoadWAVFile( "wavdata/Footsteps.wav", format, data, size, freq, loop); al.alBufferData(buffer[0], format[0], data[0], size[0], freq[0]); al.alGenSources(1, source, 0); al.alSourcei(source[0], AL.AL_BUFFER, buffer[0]); al.alSourcef(source[0], AL.AL_PITCH, 1.0f); al.alSourcef(source[0], AL.AL_GAIN, 1.0f); al.alSourcefv(source[0], AL.AL_POSITION, sourcePos, 0); al.alSourcefv(source[0], AL. AL_VELOCITY, sourceVel, 0); //[此处的 VELOCITY在原文中被误写为POSITION,在此更正——译者注] al.alSourcei(source[0], AL.AL_LOOPING, AL.AL_TRUE); if (al.alGetError() != AL.AL_NO_ERROR) { return AL.AL_FALSE; } return AL.AL_TRUE; }
以上部分包括两个变化,首先我们更换了使用的音频文件,之后将参数AL_LOOPING置为AL_TRUE,它表示声源将会循环播放直到接收到停止命令。[如果想使用自己的Wav文件,请确保它是单声道的格式,否则OpenAL会忽略其空间位置变换而带来的声音影响——译者注]
static void setListenerValues() { al.alListenerfv(AL.AL_POSITION, listenerPos, 0); al.alListenerfv(AL.AL_VELOCITY, listenerVel, 0); al.alListenerfv(AL.AL_ORIENTATION, listenerOri, 0); } static void killALData() { al.alDeleteBuffers(1, buffer, 0); al.alDeleteSources(1, source, 0); ALut.alutExit(); }
上面这部分没什么变化。
public static void main(String[] args) { ALut.alutInit(); al = ALFactory.getAL(); if(loadALData() == AL.AL_FALSE) { System.exit(1); }; setListenerValues(); al.alSourcePlay(source[0]); long startTime = System.currentTimeMillis(); long elapsed = 0; long ticker = 0; long lastTime = 0; while (elapsed < 10000) { elapsed = System.currentTimeMillis() - startTime; if (ticker > 100) { ticker = 0; sourcePos[0] += sourceVel[0]; sourcePos[1] += sourceVel[1]; sourcePos[2] += sourceVel[2]; al.alSourcefv( source[0], AL.AL_POSITION, sourcePos, 0); } ticker += System.currentTimeMillis() - lastTime; lastTime = System.currentTimeMillis(); } ALut.alutExit(); } }
上面这部分与之前的相比,我们改变了它的循环结构。声音不会突然停止,而是随着声源与听众距离的增加而缓慢衰减,我们不断向位置量累加其对应速度来达到这一效果,所用的时间由系统的时间函数计算获得。上面这些时间量基本不用修改,但如果音频淡出过快的话你可以将上面的100改为更高的数字。
时间: 2024-10-07 14:06:43