axmol/CocosDenshion/SimpleAudioEngine/SimpleAudioEngine.cpp

465 lines
10 KiB
C++

#include "SimpleAudioEngine.h"
#include "TG3.h"
#include <map>
#include <string>
#define BREAK_IF(cond) if (cond) break;
unsigned int BKDRHash(const char *str)
{
unsigned int seed = 31; // 31 131 1313 13131 131313 etc..
unsigned int hash = 0;
while (*str)
{
hash = hash * seed + (*str++);
}
return (hash & 0x7FFFFFFF);
}
static SimpleAudioEngine s_SharedAudioEngie;
static SoundResHandle s_HSoundRes;
typedef std::map<std::string, int> SoundInfoMap;
static SoundInfoMap s_SoundMap;
SimpleAudioEngine::SimpleAudioEngine()
: m_nBackgroundMusicVolume(100)
, m_nEffectsVolume(100)
, m_bWillPlayBackgroundMusic(false)
, m_pEffects(NULL)
{
m_pEffectPlayers = new PlayerArray();
m_pBackPlayer = new SoundPlayer();
}
SimpleAudioEngine::~SimpleAudioEngine()
{
removeAllEffects();
removeAllEffectPlayers();
if (m_pBackPlayer)
{
delete m_pBackPlayer;
m_pBackPlayer = NULL;
}
}
SimpleAudioEngine* SimpleAudioEngine::getSharedEngine()
{
return &s_SharedAudioEngie;
}
void SimpleAudioEngine::playBackgroundMusic(const char* pszFilePath, bool bLoop)
{
int nTimes = 1;
if (bLoop)
{
nTimes = -1;
}
if (FileUtils::isFileExisted(pszFilePath))
{
if (m_pBackPlayer)
{
m_pBackPlayer->PlaySoundFile(pszFilePath, nTimes);
}
}
else
{
do
{
SoundInfoMap::iterator iter;
iter = s_SoundMap.find(pszFilePath);
BREAK_IF(iter == s_SoundMap.end());
unsigned int nSize = 0;
const void* pData = s_HSoundRes.LoadConstRawData(iter->second, &nSize);
BREAK_IF(!pData);
if (m_pBackPlayer)
{
m_pBackPlayer->PlaySoundFromMem((unsigned char*)pData, nSize, pszFilePath, nTimes);
}
} while (0);
}
}
void SimpleAudioEngine::stopBackgroundMusic()
{
if (m_pBackPlayer)
{
m_pBackPlayer->Stop();
}
}
void SimpleAudioEngine::pauseBackgroundMusic()
{
if (m_pBackPlayer)
{
m_pBackPlayer->Pause();
}
}
void SimpleAudioEngine::resumeBackgroundMusic()
{
if (m_pBackPlayer)
{
m_pBackPlayer->Resume();
}
}
void SimpleAudioEngine::rewindBackgroundMusic()
{
if (m_pBackPlayer)
{
m_pBackPlayer->Rewind();
}
}
bool SimpleAudioEngine::willPlayBackgroundMusic()
{
return m_bWillPlayBackgroundMusic;
}
bool SimpleAudioEngine::isBackgroundMusicPlaying()
{
bool bRet = false;
if (m_pBackPlayer)
{
bRet = m_pBackPlayer->IsPlaying();
}
return bRet;
}
// properties
int SimpleAudioEngine::GetBackgroundMusicVolume()
{
return m_nBackgroundMusicVolume;
}
void SimpleAudioEngine::SetBackgroundMusicVolume(int volume)
{
if (m_pBackPlayer)
{
m_pBackPlayer->SetVolumeValue(volume);
}
m_nBackgroundMusicVolume = volume;
}
int SimpleAudioEngine::GetEffectsVolume()
{
return m_nEffectsVolume;
}
void SimpleAudioEngine::SetEffectsVolume(int volume)
{
PlayerArrayIterator iter;
for (iter = m_pEffectPlayers->begin(); iter != m_pEffectPlayers->end(); ++iter)
{
if (*iter)
{
(*iter)->SetVolumeValue(volume);
}
}
m_nEffectsVolume = volume;
}
// for sound effects
int SimpleAudioEngine::playEffect(const char* pszFilePath)
{
int nSoundID = preloadEffect(pszFilePath);
if (nSoundID > 0)
{
playPreloadedEffect(nSoundID);
}
return nSoundID;
}
void SimpleAudioEngine::stopEffect(int nSoundId)
{
SoundPlayer* pPlayer = NULL;
// find the players are playing the effect
PlayerArrayIterator iter;
for (iter = m_pEffectPlayers->begin(); iter != m_pEffectPlayers->end(); ++iter)
{
if ((*iter) && pPlayer->GetCurrentSoundID() == nSoundId)
{
pPlayer->Stop();
}
}
}
int SimpleAudioEngine::preloadEffect(const char* pszFilePath)
{
int nSoundID = 0;
if (! FileUtils::isFileExisted(pszFilePath))
{
// if the file is not existed, find in the ResourceInfo
nSoundID = loadFromResourceInfo(pszFilePath);
}
else
{
// load effect info from file
nSoundID = loadFromFile(pszFilePath);
}
return nSoundID;
}
void SimpleAudioEngine::unloadEffect(int nSoundId)
{
do
{
tHashElement* pElement = NULL;
HASH_FIND_INT(m_pEffects, &nSoundId, pElement);
BREAK_IF(!pElement);
delete [] (pElement->pDataBuffer);
HASH_DEL(m_pEffects, pElement);
free(pElement);
} while (0);
}
void SimpleAudioEngine::playPreloadedEffect(int nSoundId)
{
do
{
tHashElement* pElement = NULL;
HASH_FIND_INT(m_pEffects, &nSoundId, pElement);
BREAK_IF(!pElement);
SoundPlayer* pPlayer = NULL;
bool bLoaded = false;
// find the not playing player in m_pEffectPlayers
PlayerArrayIterator iter;
for (iter = m_pEffectPlayers->begin(); iter != m_pEffectPlayers->end(); ++iter)
{
if ((*iter) && !(*iter)->IsPlaying())
{
pPlayer = (*iter);
if (pPlayer->GetCurrentSoundID() == nSoundId)
{
bLoaded = true;
break;
}
}
}
// not find player,new one player
if (!pPlayer)
{
pPlayer = new SoundPlayer();
m_pEffectPlayers->push_back(pPlayer);
// set the player volume
pPlayer->SetVolumeValue(m_nEffectsVolume);
}
// play the sound and record the player
if (bLoaded)
{
pPlayer->Rewind();
}
else
{
pPlayer->PlaySoundFromMem(pElement->pDataBuffer, pElement->nDataSize, pElement->FileName);
pPlayer->SetCurrentSoundID(nSoundId);
}
} while (0);
}
void SimpleAudioEngine::removeAllEffects()
{
for (tHashElement *pElement = m_pEffects; pElement != NULL; )
{
int nSoundID = pElement->nSoundID;
pElement = (tHashElement*)pElement->hh.next;
unloadEffect(nSoundID);
}
}
void SimpleAudioEngine::removeAllEffectPlayers()
{
PlayerArrayIterator iter;
for (iter = m_pEffectPlayers->begin(); iter != m_pEffectPlayers->end(); ++iter)
{
if (*iter)
{
delete *iter;
}
}
m_pEffectPlayers->clear();
delete m_pEffectPlayers;
}
int SimpleAudioEngine::loadFromResourceInfo(const char* pFileKey)
{
SoundInfoMap::iterator iter;
iter = s_SoundMap.find(pFileKey);
int nSoundID = 0;
do
{
BREAK_IF(iter == s_SoundMap.end());
// if we have loaded the file before,break
tHashElement *pElement = NULL;
HASH_FIND_INT(m_pEffects, &iter->second, pElement);
if (pElement)
{
nSoundID = iter->second;
break;
}
unsigned int nSize = 0;
const void* pData = s_HSoundRes.LoadConstRawData(iter->second, &nSize);
BREAK_IF(!pData);
// copy the data
unsigned char* pSoundData = new unsigned char[nSize];
MemCopy(pSoundData, pData, nSize);
// record the id
nSoundID = iter->second;
// add the data to hash map
pElement = (tHashElement*)calloc(sizeof(*pElement), 1);
pElement->nSoundID = nSoundID;
pElement->pDataBuffer = pSoundData;
pElement->nDataSize = nSize;
pElement->FileName = pFileKey;
HASH_ADD_INT(m_pEffects, nSoundID, pElement);
} while (0);
return nSoundID;
}
int SimpleAudioEngine::loadFromFile(const char* pFilePath)
{
int nSoundID = 0;
do
{
int nID = BKDRHash(pFilePath);
// if we have loaded the file before,break
tHashElement *pElement = NULL;
HASH_FIND_INT(m_pEffects, &nID, pElement);
if (pElement)
{
nSoundID = nID;
break;
}
// calculate the buffer size we needed
SoundPlayer TempPlayer;
int nBufferSize = TempPlayer.GetFileBufferSize(pFilePath);
// can not calculate the size,load failed
BREAK_IF(nBufferSize < 0);
// load the file data
unsigned char* buffer = NULL;
buffer = new unsigned char[nBufferSize];
BREAK_IF(!buffer);
int nSize = TempPlayer.DecodeFile(buffer, nBufferSize, pFilePath);
BREAK_IF(nSize < 0);
// record the id
nSoundID = nID;
// add the data to hash map
pElement = (tHashElement*)calloc(sizeof(*pElement), 1);
pElement->nSoundID = nSoundID;
pElement->pDataBuffer = buffer;
pElement->nDataSize = nBufferSize;
pElement->FileName = "";
HASH_ADD_INT(m_pEffects, nSoundID, pElement);
} while (0);
return nSoundID;
}
void SimpleAudioEngine::setSoundResInfo(const T_SoundResInfo ResInfo[], int nCount)
{
// first, clear the map before
if (!s_SoundMap.empty())
{
s_SoundMap.clear();
}
// second, insert the pairs
for (int i = 0; i < nCount; ++i)
{
std::string name = (ResInfo[i]).FileName;
int nResID = (ResInfo[i]).nResID;
s_SoundMap.insert(SoundInfoMap::value_type(name, nResID));
}
}
void SimpleAudioEngine::setResourceEntry(const AppResourceEntry* pResEntry)
{
if (pResEntry)
{
s_HSoundRes.setResourceEntry(pResEntry);
}
}
//////////////////////////////////////////////////
//
// SoundResHandle
//
//////////////////////////////////////////////////
SoundResHandle::SoundResHandle()
:m_pResLib(NULL)
{
}
SoundResHandle::~SoundResHandle()
{
release();
}
void SoundResHandle::release()
{
if (m_pResLib)
{
delete m_pResLib;
m_pResLib = NULL;
}
}
void SoundResHandle::setResourceEntry(const AppResourceEntry* pResEntry)
{
release();
m_pResLib = new TResourceLib(pResEntry);
}
const void* SoundResHandle::LoadConstRawData(int nResID, unsigned int* nLen)
{
const void* pResult = NULL;
if (m_pResLib)
{
pResult = m_pResLib->LoadConstRawData(nResID, nLen);
}
return pResult;
}