axmol/CocosDenshion/bada/SimpleAudioEngine.cpp

485 lines
11 KiB
C++

#include "SimpleAudioEngine.h"
#include "CCAudioOut.h"
#include <FBase.h>
#include <FMedia.h>
#include <FSystem.h>
#include <string>
#include <map>
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
using namespace std;
using namespace Osp::Media;
using namespace Osp::System;
using namespace Osp::Base::Runtime;
using namespace Osp::Base::Collection;
#define BREAK_IF(cond) if (cond) break;
namespace CocosDenshion {
#define MAX_BUFFER_SIZE 2520 // 840 byte
typedef map<unsigned int, CCAudioOut*> EffectList;
typedef pair<unsigned int ,CCAudioOut*> Effect;
static SimpleAudioEngine *s_pSharedAudioEngine = NULL;
static Player *s_pBackPlayer = NULL;
static EffectList s_List;
static float s_fBackgroundMusicVolume = 1.0f;
static float s_fEffectsVolume = 1.0f;
static bool s_bWillPlayBackgroundMusic = false;
static string s_strResourcePath = "/Res/";
static unsigned int _Hash(const char *key)
{
unsigned int len = strlen(key);
const char *end=key+len;
unsigned int hash;
for (hash = 0; key < end; key++)
{
hash *= 16777619;
hash ^= (unsigned int) (unsigned char) toupper(*key);
}
return (hash);
}
class MyPlayerEventListener :
public IPlayerEventListener
{
public:
/**
* Notifies that audio/video content was opened asynchronously.
*
* @param[in] r The cause of the error
* @exception E_SUCCESS The method was successful.
* @exception E_SYSTEM A system error occurred.
* @exception E_CONNECTION_FAILED Network connection failed.
* @exception E_UNSUPPORTED_FORMAT The specified format is not supported.
* @exception E_UNSUPPORTED_CODEC The specified codec is not supported.
* @exception E_OUT_OF_MEMORY Insufficient memory.
* @see Player::OpenFile(), Player::OpenUrl(), Player::OpenBuffer()
*/
virtual void OnPlayerOpened( result r )
{
AppLog("OnPlayerOpened");
}
/**
* Notifies that the Player has reached the end of the clip.
*
*/
virtual void OnPlayerEndOfClip(void)
{
AppLog("OnPlayerEndOfClip");
}
/**
* Notifies that the position of the audio/video content was moved asynchronously.
*
* @exception E_SUCCESS The method was successful.
* @exception E_SYSTEM A system error occurred.
* @see Player::SeekTo()
*/
virtual void OnPlayerSeekCompleted( result r )
{
AppLog("OnPlayerSeekCompleted");
}
/**
* Notifies that streaming data is being buffered.
*
* @param[in] percent The percentage of buffering completed
* @see Player::OpenUrl()
*/
virtual void OnPlayerBuffering(int percent)
{
AppLog("OnPlayerBuffering");
}
/**
* Notifies that an error has occurred while the Player is working.
*
* @param[in] r A player error reason of type ::PlayerErrorReason
* @remark While playing streaming media, the player might throw an error like ::PLAYER_ERROR_CONNECTION_LOST @n
* ::PLAYER_ERROR_STREAMING_TIMEOUT, ::PLAYER_ERROR_TRANSPORT or ::PLAYER_ERROR_SERVER. @n
* If the content includes invalid data, ::PLAYER_ERROR_INVALID_DATA may occur.
* @see PlayerErrorReason
*/
virtual void OnPlayerErrorOccurred( PlayerErrorReason r )
{
AppLog("OnPlayerErrorOccurred");
}
/**
* Notifies that the Player is being interrupted by a task of higher priority than Player.
*
*/
virtual void OnPlayerInterrupted(void)
{
//Insert your code here
AppLog("OnPlayerInterrupted");
if (s_pBackPlayer->GetState() == PLAYER_STATE_PLAYING)
s_pBackPlayer->Pause();
}
/**
* Notifies that the interrupting Player has been released.
*
*/
virtual void OnPlayerReleased(void)
{
//Insert your code here
AppLog("OnPlayerReleased");
if (s_pBackPlayer->GetState() != PLAYER_STATE_PLAYING)
s_pBackPlayer->Play();
}
};
static MyPlayerEventListener s_playerListener;
static void closeMediaPlayer(Player*& pPlayer)
{
if (pPlayer != NULL)
{
PlayerState nowState = pPlayer->GetState();
if( nowState == PLAYER_STATE_PLAYING || nowState == PLAYER_STATE_PAUSED )
{
pPlayer->Stop();
pPlayer->Close();
AppLog("audio player closed");
}
else if(nowState == PLAYER_STATE_OPENED || nowState == PLAYER_STATE_ENDOFCLIP || nowState == PLAYER_STATE_STOPPED )
{
pPlayer->Close();
AppLog("audio player closed");
}
}
}
static bool openMediaPlayer(Player*& pPlayer, const char* pszFilePath, bool bLoop)
{
bool bRet = false;
result r = E_FAILURE;
do
{
closeMediaPlayer(pPlayer);
if (pPlayer == NULL)
{
pPlayer = new Player();
r = pPlayer->Construct(s_playerListener, null);
if (IsFailed(r))
{
AppLog("player construct fails, pszFilePath = %s", pszFilePath);
delete pPlayer;
pPlayer = NULL;
break;
}
}
string strFilePath = s_strResourcePath+pszFilePath;
// OpenFile must use synchronous param, for after that it will playing.
r = pPlayer->OpenFile(strFilePath.c_str(), false);
if (IsFailed(r))
{
AppLog("Open (%s) fails\n", strFilePath.c_str());
delete pPlayer;
pPlayer = NULL;
break;
}
else
{
pPlayer->SetLooping(bLoop);
bRet = true;
}
}
while (0);
return bRet;
}
SimpleAudioEngine::SimpleAudioEngine()
{
}
SimpleAudioEngine::~SimpleAudioEngine()
{
AppLog("destroy SimpleAudioEngine");
for (EffectList::iterator it = s_List.begin(); it != s_List.end(); ++it)
{
it->second->Reset();
delete it->second;
}
s_List.clear();
closeMediaPlayer(s_pBackPlayer);
delete s_pBackPlayer;
s_pBackPlayer = NULL;
}
SimpleAudioEngine* SimpleAudioEngine::sharedEngine()
{
if (s_pSharedAudioEngine == NULL)
{
s_pSharedAudioEngine = new SimpleAudioEngine;
}
return s_pSharedAudioEngine;
}
void SimpleAudioEngine::end()
{
if (s_pSharedAudioEngine)
{
delete s_pSharedAudioEngine;
s_pSharedAudioEngine = NULL;
}
}
void SimpleAudioEngine::setResource(const char* pszZipFileName)
{
}
void SimpleAudioEngine::setResourcePath(const char *pszResourcePath)
{
s_strResourcePath = pszResourcePath;
}
void SimpleAudioEngine::preloadBackgroundMusic(const char* pszFilePath)
{
}
void SimpleAudioEngine::playBackgroundMusic(const char* pszFilePath, bool bLoop)
{
result r = E_FAILURE;
bool bRet = false;
bRet = openMediaPlayer(s_pBackPlayer, pszFilePath, bLoop);
setBackgroundMusicVolume(s_fBackgroundMusicVolume);
if (bRet && s_fBackgroundMusicVolume > 0.0f)
{
r = s_pBackPlayer->Play();
}
}
void SimpleAudioEngine::stopBackgroundMusic(bool bReleaseData)
{
if (s_pBackPlayer && PLAYER_STATE_PLAYING == s_pBackPlayer->GetState())
{
s_pBackPlayer->Stop();
}
}
void SimpleAudioEngine::pauseBackgroundMusic()
{
if (s_pBackPlayer && PLAYER_STATE_PLAYING == s_pBackPlayer->GetState())
{
s_pBackPlayer->Pause();
}
}
void SimpleAudioEngine::resumeBackgroundMusic()
{
if (s_pBackPlayer && PLAYER_STATE_PLAYING != s_pBackPlayer->GetState())
{
s_pBackPlayer->Play();
}
}
void SimpleAudioEngine::rewindBackgroundMusic()
{
if (s_pBackPlayer)
{
s_pBackPlayer->SeekTo(0);
if (PLAYER_STATE_PLAYING != s_pBackPlayer->GetState())
{
s_pBackPlayer->Play();
}
}
}
bool SimpleAudioEngine::willPlayBackgroundMusic()
{
return s_bWillPlayBackgroundMusic;
}
bool SimpleAudioEngine::isBackgroundMusicPlaying()
{
bool bRet = false;
if (s_pBackPlayer)
{
if (s_pBackPlayer->GetState() == PLAYER_STATE_PLAYING)
{
bRet = true;
}
}
return bRet;
}
// properties
float SimpleAudioEngine::getBackgroundMusicVolume()
{
return s_fBackgroundMusicVolume;
}
void SimpleAudioEngine::setBackgroundMusicVolume(float volume)
{
if (volume > 1.0f)
{
volume = 1.0f;
}
else if (volume < 0.0f)
{
volume = 0.0f;
}
if (s_pBackPlayer)
{
s_pBackPlayer->SetVolume((int) (volume * 99));
if (volume > 0.0f && s_pBackPlayer->GetVolume() == 0)
{
AppLog("volume is lowest");
s_pBackPlayer->SetVolume(1);
}
}
AppLog("volume = %f", volume);
s_fBackgroundMusicVolume = volume;
}
float SimpleAudioEngine::getEffectsVolume()
{
return s_fEffectsVolume;
}
void SimpleAudioEngine::setEffectsVolume(float volume)
{
if (volume > 1.0f)
{
volume = 1.0f;
}
else if (volume < 0.0f)
{
volume = 0.0f;
}
s_fEffectsVolume = volume;
}
// for sound effects
unsigned int SimpleAudioEngine::playEffect(const char* pszFilePath, bool bLoop/* = false*/)
{
long long curTick, oldTick;
SystemTime::GetTicks(oldTick);
result r = E_FAILURE;
string strFilePath = s_strResourcePath+pszFilePath;
unsigned int nRet = _Hash(strFilePath.c_str());
AppLog("play effect (%s)", pszFilePath);
preloadEffect(pszFilePath);
EffectList::iterator p = s_List.find(nRet);
if (p != s_List.end())
{
p->second->SetVolume((int) (s_fEffectsVolume * 99));
int volume = p->second->GetVolume();
AppLog("volume = %d, s_fEffectsVolume = %f", volume, s_fEffectsVolume);
if (s_fEffectsVolume > 0.0f && volume == 0)
{
AppLog("effect volume is lowest");
p->second->SetVolume(1);
}
if (AUDIOOUT_STATE_PLAYING == p->second->GetState())
{
AppLog("Reset effect...");
r = p->second->Reset();
}
AppLog("play...");
r = p->second->Play();
if (IsFailed(r))
{
AppLog("play effect fails, error code = %d", r);
}
}
SystemTime::GetTicks(curTick);
AppLog("play effect waste %ld ms...", (long)(curTick-oldTick));
return nRet;
}
void SimpleAudioEngine::stopEffect(unsigned int nSoundId)
{
CCAudioOut*& pPlayer = s_List[nSoundId];
if (pPlayer != NULL)
{
pPlayer->Reset();
}
}
void SimpleAudioEngine::preloadEffect(const char* pszFilePath)
{
int nRet = 0;
CCAudioOut* pEffectPlayer = NULL;
do
{
BREAK_IF(! pszFilePath);
string strFilePath = s_strResourcePath+pszFilePath;
nRet = _Hash(strFilePath.c_str());
BREAK_IF(s_List.end() != s_List.find(nRet));
AppLog("not find effect, create it...");
// bada only support 10 player instance, we use one for background music, other for effect music.
if (s_List.size() >= 64)
{
// get the first effect, and remove it form list
AppLog("effect preload more than 9, delete the first effect");
pEffectPlayer = s_List.begin()->second;
// closeMediaPlayer(pEffectPlayer);
pEffectPlayer->Finalize();
s_List.erase(s_List.begin()->first);
}
if (pEffectPlayer == NULL)
pEffectPlayer = new CCAudioOut;
pEffectPlayer->Initialize(strFilePath.c_str());
// if (openMediaPlayer(pEffectPlayer, pszFilePath, false))
{
s_List.insert(Effect(nRet, pEffectPlayer));
}
} while (0);
}
void SimpleAudioEngine::unloadEffect(const char* pszFilePath)
{
string strFilePath = s_strResourcePath+pszFilePath;
unsigned int nSoundId = _Hash(strFilePath.c_str());
CCAudioOut*& pPlayer = s_List[nSoundId];
pPlayer->Reset();
//closeMediaPlayer(pPlayer);
}
} // end of namespace CocosDenshion