// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdsound.h" #include "sound.h" #include "nel/misc/path.h" #include "sound_bank.h" #include "simple_sound.h" #include "complex_sound.h" #include "background_sound.h" #include "context_sound.h" #include "music_sound.h" #include "stream_sound.h" using namespace std; using namespace NLMISC; namespace NLSOUND { CSound *CSound::createSound(const std::string &filename, NLGEORGES::UFormElm& formRoot) { CSound *ret = NULL; string soundType; NLGEORGES::UFormElm *psoundType; if (!formRoot.getNodeByName(&psoundType, ".SoundType")) { nlwarning("No SoundType in : %s", filename.c_str()); return 0; } if (psoundType != NULL) { std::string dfnName; psoundType->getDfnName(dfnName); if (dfnName == "simple_sound.dfn") { ret = new CSimpleSound(); ret->importForm(filename, formRoot); } else if (dfnName == "complex_sound.dfn") { ret = new CComplexSound(); ret->importForm(filename, formRoot); } else if (dfnName == "background_sound.dfn") { ret = new CBackgroundSound(); ret->importForm(filename, formRoot); } else if (dfnName == "context_sound.dfn") { ret = new CContextSound(); ret->importForm(filename, formRoot); } else if (dfnName == "music_sound.dfn") { ret = new CMusicSound(); ret->importForm(filename, formRoot); } else if (dfnName == "stream_sound.dfn") { ret = new CStreamSound(); ret->importForm(filename, formRoot); } else { nlassertex(false, ("SoundType unsuported : %s", dfnName.c_str())); } } return ret; } /* * Constructor */ CSound::CSound() : _Gain(1.0f), _Pitch(1.0f), _Priority(MidPri), _ConeInnerAngle(6.283185f), _ConeOuterAngle(6.283185f), _ConeOuterGain( 1.0f ), _Looping(false), _MinDist(1.0f), _MaxDist(1000000.0f), _UserVarControler(CStringMapper::emptyId()) { } CSound::~CSound() {} void CSound::serial(NLMISC::IStream &s) { s.serial(_Gain); s.serial(_Pitch); s.serialEnum(_Priority); s.serial(_ConeInnerAngle, _ConeOuterAngle, _ConeOuterGain); s.serial(_Direction); s.serial(_Looping); s.serial(_MaxDist); if (s.isReading()) { std::string name; s.serial(name); _Name = CStringMapper::map(name); } else { std::string name = CStringMapper::unmap(_Name); s.serial(name); } } /** * Load the sound parameters from georges' form */ void CSound::importForm(const std::string& filename, NLGEORGES::UFormElm& root) { // Name _Name = CStringMapper::map(CFile::getFilenameWithoutExtension(filename)); // InternalConeAngle uint32 inner; root.getValueByName(inner, ".InternalConeAngle"); if (inner > 360) { inner = 360; } _ConeInnerAngle = (float) (Pi * inner / 180.0f); // convert to radians // ExternalConeAngle uint32 outer; root.getValueByName(outer, ".ExternalConeAngle"); if (outer > 360) { outer = 360; } _ConeOuterAngle= (float) (Pi * outer / 180.0f); // convert to radians // Loop root.getValueByName(_Looping, ".Loop"); // Gain sint32 gain; root.getValueByName(gain, ".Gain"); if (gain > 0) { gain = 0; } if (gain < -100) { gain = -100; } _Gain = (float) pow(10.0, gain / 20.0); // convert dB to linear gain // External gain root.getValueByName(gain, ".ExternalGain"); if (gain > 0) { gain = 0; } if (gain < -100) { gain = -100; } _ConeOuterGain = (float) pow(10.0, gain / 20.0); // convert dB to linear gain // Direction float x, y, z; root.getValueByName(x, ".Direction.X"); root.getValueByName(y, ".Direction.Y"); root.getValueByName(z, ".Direction.Z"); _Direction = CVector(x, y, z); // Pitch sint32 trans; root.getValueByName(trans, ".Transpose"); _Pitch = (float) pow(Sqrt12_2, trans); // convert semi-tones to playback speed // Priority uint32 prio = 0; root.getValueByName(prio, ".Priority"); switch (prio) { case 0: _Priority = LowPri; break; case 1: _Priority = MidPri; break; case 2: _Priority = HighPri; break; case 3: _Priority = HighestPri; break; default: _Priority = MidPri; } } } // NLSOUND