#include "SimpleAudioEngine.h" #include "TG3.h" #include #include #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 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; }