2015-11-25 09:54:12 +08:00
|
|
|
/**
|
|
|
|
* @author cesarpachon
|
2016-05-19 02:34:38 +08:00
|
|
|
*/
|
2015-11-25 09:54:12 +08:00
|
|
|
#include <cstring>
|
2016-05-20 00:39:40 +08:00
|
|
|
#include <cstdint>
|
2016-03-20 21:53:44 +08:00
|
|
|
#include "audio/linux/AudioEngine-linux.h"
|
2016-04-18 15:09:21 +08:00
|
|
|
|
|
|
|
#include "base/CCDirector.h"
|
|
|
|
#include "base/CCScheduler.h"
|
|
|
|
#include "platform/CCFileUtils.h"
|
|
|
|
|
2015-11-25 09:54:12 +08:00
|
|
|
using namespace cocos2d;
|
|
|
|
using namespace cocos2d::experimental;
|
|
|
|
|
|
|
|
AudioEngineImpl * g_AudioEngineImpl = nullptr;
|
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void ERRCHECKWITHEXIT(FMOD_RESULT result)
|
|
|
|
{
|
2015-11-25 09:54:12 +08:00
|
|
|
if (result != FMOD_OK) {
|
|
|
|
printf("FMOD error! (%d) %s\n", result, FMOD_ErrorString(result));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool ERRCHECK(FMOD_RESULT result)
|
|
|
|
{
|
2015-11-25 09:54:12 +08:00
|
|
|
if (result != FMOD_OK) {
|
|
|
|
printf("FMOD error! (%d) %s\n", result, FMOD_ErrorString(result));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-05-19 02:34:38 +08:00
|
|
|
FMOD_RESULT F_CALLBACK channelCallback(FMOD_CHANNELCONTROL *channelcontrol,
|
|
|
|
FMOD_CHANNELCONTROL_TYPE controltype,
|
|
|
|
FMOD_CHANNELCONTROL_CALLBACK_TYPE callbacktype,
|
2015-11-25 09:54:12 +08:00
|
|
|
void *commandData1, void *commandData2)
|
|
|
|
{
|
2016-05-19 04:17:33 +08:00
|
|
|
if (controltype == FMOD_CHANNELCONTROL_CHANNEL && callbacktype == FMOD_CHANNELCONTROL_CALLBACK_END) {
|
2015-11-25 09:54:12 +08:00
|
|
|
g_AudioEngineImpl->onSoundFinished((FMOD::Channel *)channelcontrol);
|
|
|
|
}
|
|
|
|
return FMOD_OK;
|
|
|
|
}
|
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
AudioEngineImpl::AudioEngineImpl()
|
|
|
|
{
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
AudioEngineImpl::~AudioEngineImpl()
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD_RESULT result;
|
|
|
|
result = pSystem->close();
|
|
|
|
ERRCHECKWITHEXIT(result);
|
|
|
|
result = pSystem->release();
|
|
|
|
ERRCHECKWITHEXIT(result);
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool AudioEngineImpl::init()
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD_RESULT result;
|
|
|
|
/*
|
|
|
|
Create a System object and initialize.
|
|
|
|
*/
|
|
|
|
result = FMOD::System_Create(&pSystem);
|
|
|
|
ERRCHECKWITHEXIT(result);
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
result = pSystem->setOutput(FMOD_OUTPUTTYPE_PULSEAUDIO);
|
|
|
|
ERRCHECKWITHEXIT(result);
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
result = pSystem->init(32, FMOD_INIT_NORMAL, 0);
|
|
|
|
ERRCHECKWITHEXIT(result);
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo.clear();
|
|
|
|
mapSound.clear();
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
auto scheduler = cocos2d::Director::getInstance()->getScheduler();
|
|
|
|
scheduler->schedule(schedule_selector(AudioEngineImpl::update), this, 0.05f, false);
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
g_AudioEngineImpl = this;
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
return true;
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
int AudioEngineImpl::play2d(const std::string &fileFullPath, bool loop, float volume)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
int id = preload(fileFullPath, nullptr);
|
2016-05-19 04:17:33 +08:00
|
|
|
if (id >= 0) {
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[id].loop=loop;
|
|
|
|
mapChannelInfo[id].channel->setPaused(true);
|
|
|
|
mapChannelInfo[id].volume = volume;
|
|
|
|
AudioEngine::_audioIDInfoMap[id].state = AudioEngine::AudioState::PAUSED;
|
|
|
|
resume(id);
|
|
|
|
}
|
|
|
|
return id;
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::setVolume(int audioID, float volume)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[audioID].channel->setVolume(volume);
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::setVolume: invalid audioID: %d\n", audioID);
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::setLoop(int audioID, bool loop)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
mapChannelInfo[audioID].channel->setLoopCount(loop ? -1 : 0);
|
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::setLoop: invalid audioID: %d\n", audioID);
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool AudioEngineImpl::pause(int audioID)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[audioID].channel->setPaused(true);
|
|
|
|
AudioEngine::_audioIDInfoMap[audioID].state = AudioEngine::AudioState::PAUSED;
|
|
|
|
return true;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::pause: invalid audioID: %d\n", audioID);
|
|
|
|
return false;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool AudioEngineImpl::resume(int audioID)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
if (!mapChannelInfo[audioID].channel) {
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD::Channel *channel = nullptr;
|
|
|
|
FMOD::ChannelGroup *channelgroup = nullptr;
|
|
|
|
//starts the sound in pause mode, use the channel to unpause
|
|
|
|
FMOD_RESULT result = pSystem->playSound(mapChannelInfo[audioID].sound, channelgroup, true, &channel);
|
2016-05-19 04:17:33 +08:00
|
|
|
if (ERRCHECK(result)) {
|
2016-05-19 02:34:04 +08:00
|
|
|
return false;
|
|
|
|
}
|
2016-05-19 04:17:33 +08:00
|
|
|
channel->setMode(mapChannelInfo[audioID].loop ? FMOD_LOOP_NORMAL : FMOD_LOOP_OFF);
|
|
|
|
channel->setLoopCount(mapChannelInfo[audioID].loop ? -1 : 0);
|
2016-05-19 02:34:04 +08:00
|
|
|
channel->setVolume(mapChannelInfo[audioID].volume);
|
2016-05-20 00:39:40 +08:00
|
|
|
channel->setUserData(reinterpret_cast<void *>(static_cast<std::intptr_t>(mapChannelInfo[audioID].id)));
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[audioID].channel = channel;
|
|
|
|
}
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[audioID].channel->setPaused(false);
|
|
|
|
AudioEngine::_audioIDInfoMap[audioID].state = AudioEngine::AudioState::PLAYING;
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
return true;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::resume: invalid audioID: %d\n", audioID);
|
|
|
|
return false;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool AudioEngineImpl::stop(int audioID)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[audioID].channel->stop();
|
|
|
|
mapChannelInfo[audioID].channel = nullptr;
|
|
|
|
return true;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::stop: invalid audioID: %d\n", audioID);
|
|
|
|
return false;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::stopAll()
|
|
|
|
{
|
2016-10-27 15:10:24 +08:00
|
|
|
for (auto& it : mapChannelInfo) {
|
|
|
|
ChannelInfo & audioRef = it.second;
|
2016-05-19 02:34:04 +08:00
|
|
|
audioRef.channel->stop();
|
|
|
|
audioRef.channel = nullptr;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
float AudioEngineImpl::getDuration(int audioID)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD::Sound * sound = mapChannelInfo[audioID].sound;
|
|
|
|
unsigned int length;
|
|
|
|
FMOD_RESULT result = sound->getLength(&length, FMOD_TIMEUNIT_MS);
|
|
|
|
ERRCHECK(result);
|
|
|
|
float duration = (float)length / 1000.0f;
|
|
|
|
return duration;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::getDuration: invalid audioID: %d\n", audioID);
|
|
|
|
return AudioEngine::TIME_UNKNOWN;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
float AudioEngineImpl::getCurrentTime(int audioID)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
unsigned int position;
|
|
|
|
FMOD_RESULT result = mapChannelInfo[audioID].channel->getPosition(&position, FMOD_TIMEUNIT_MS);
|
|
|
|
ERRCHECK(result);
|
|
|
|
float currenttime = position /1000.0f;
|
|
|
|
return currenttime;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::getCurrentTime: invalid audioID: %d\n", audioID);
|
|
|
|
return AudioEngine::TIME_UNKNOWN;
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
bool AudioEngineImpl::setCurrentTime(int audioID, float time)
|
|
|
|
{
|
2016-09-20 14:50:47 +08:00
|
|
|
bool ret = false;
|
2016-05-19 04:17:33 +08:00
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
unsigned int position = (unsigned int)(time * 1000.0f);
|
|
|
|
FMOD_RESULT result = mapChannelInfo[audioID].channel->setPosition(position, FMOD_TIMEUNIT_MS);
|
2016-09-20 14:50:47 +08:00
|
|
|
ret = !ERRCHECK(result);
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::setCurrentTime: invalid audioID: %d\n", audioID);
|
|
|
|
}
|
2016-09-20 14:50:47 +08:00
|
|
|
return ret;
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::setFinishCallback(int audioID, const std::function<void (int, const std::string &)> &callback)
|
|
|
|
{
|
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD::Channel * channel = mapChannelInfo[audioID].channel;
|
|
|
|
mapChannelInfo[audioID].callback = callback;
|
|
|
|
FMOD_RESULT result = channel->setCallback(channelCallback);
|
|
|
|
ERRCHECK(result);
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::setFinishCallback: invalid audioID: %d\n", audioID);
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::onSoundFinished(FMOD::Channel * channel)
|
|
|
|
{
|
2016-05-20 00:39:40 +08:00
|
|
|
int id = 0;
|
2016-05-19 04:17:33 +08:00
|
|
|
try {
|
2016-05-19 02:34:04 +08:00
|
|
|
void * data;
|
|
|
|
channel->getUserData(&data);
|
2016-05-20 00:39:40 +08:00
|
|
|
id = static_cast<int>(reinterpret_cast<std::intptr_t>(data));
|
2016-05-19 04:17:33 +08:00
|
|
|
if (mapChannelInfo[id].callback) {
|
2016-05-19 02:34:04 +08:00
|
|
|
mapChannelInfo[id].callback(id, mapChannelInfo[id].path);
|
|
|
|
}
|
|
|
|
mapChannelInfo[id].channel = nullptr;
|
2016-05-19 04:17:33 +08:00
|
|
|
}
|
|
|
|
catch (const std::out_of_range& oor) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("AudioEngineImpl::onSoundFinished: invalid audioID: %d\n", id);
|
|
|
|
}
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::uncache(const std::string& path)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
std::string fullPath = FileUtils::getInstance()->fullPathForFilename(path);
|
|
|
|
std::map<std::string, FMOD::Sound *>::const_iterator it = mapSound.find(fullPath);
|
2016-05-19 04:17:33 +08:00
|
|
|
if (it!=mapSound.end()) {
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD::Sound * sound = it->second;
|
2016-05-19 04:17:33 +08:00
|
|
|
if (sound) {
|
2016-05-19 02:34:04 +08:00
|
|
|
sound->release();
|
|
|
|
}
|
2016-05-19 02:34:38 +08:00
|
|
|
mapSound.erase(it);
|
|
|
|
}
|
2016-05-26 19:57:56 +08:00
|
|
|
if (mapId.find(path) != mapId.end())
|
|
|
|
mapId.erase(path);
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::uncacheAll()
|
|
|
|
{
|
2016-10-27 15:10:24 +08:00
|
|
|
for (const auto& it : mapSound) {
|
|
|
|
auto sound = it.second;
|
2016-05-19 04:17:33 +08:00
|
|
|
if (sound) {
|
2016-05-19 02:34:04 +08:00
|
|
|
sound->release();
|
|
|
|
}
|
2016-05-19 02:34:38 +08:00
|
|
|
}
|
|
|
|
mapSound.clear();
|
2016-05-26 19:57:56 +08:00
|
|
|
mapId.clear();
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
int AudioEngineImpl::preload(const std::string& filePath, std::function<void(bool isSuccess)> callback)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
FMOD::Sound * sound = findSound(filePath);
|
2016-05-19 04:17:33 +08:00
|
|
|
if (!sound) {
|
2016-05-19 02:34:04 +08:00
|
|
|
std::string fullPath = FileUtils::getInstance()->fullPathForFilename(filePath);
|
|
|
|
FMOD_RESULT result = pSystem->createSound(fullPath.c_str(), FMOD_LOOP_OFF, 0, &sound);
|
2016-05-19 04:17:33 +08:00
|
|
|
if (ERRCHECK(result)) {
|
2016-05-19 02:34:04 +08:00
|
|
|
printf("sound effect in %s could not be preload\n", filePath.c_str());
|
2016-05-19 04:17:33 +08:00
|
|
|
if (callback) {
|
2016-05-19 02:34:04 +08:00
|
|
|
callback(false);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
mapSound[fullPath] = sound;
|
2015-11-25 09:54:12 +08:00
|
|
|
}
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-20 00:39:40 +08:00
|
|
|
int id = static_cast<int>(mapChannelInfo.size()) + 1;
|
2016-05-26 19:57:56 +08:00
|
|
|
if (mapId.find(filePath) == mapId.end())
|
|
|
|
mapId.insert({filePath, id});
|
|
|
|
else
|
|
|
|
id = mapId.at(filePath);
|
|
|
|
|
2016-05-19 02:34:04 +08:00
|
|
|
auto& chanelInfo = mapChannelInfo[id];
|
|
|
|
chanelInfo.sound = sound;
|
2016-05-20 00:39:40 +08:00
|
|
|
chanelInfo.id = id;
|
2016-05-19 02:34:04 +08:00
|
|
|
chanelInfo.channel = nullptr;
|
|
|
|
chanelInfo.callback = nullptr;
|
|
|
|
chanelInfo.path = filePath;
|
|
|
|
//we are going to use UserData to store pointer to Channel when playing
|
2016-05-20 00:39:40 +08:00
|
|
|
chanelInfo.sound->setUserData(reinterpret_cast<void *>(static_cast<std::intptr_t>(id)));
|
2016-05-19 02:34:38 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
if (callback) {
|
2016-05-19 02:34:04 +08:00
|
|
|
callback(true);
|
|
|
|
}
|
|
|
|
return id;
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
void AudioEngineImpl::update(float dt)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
pSystem->update();
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|
2015-11-25 09:54:12 +08:00
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
FMOD::Sound * AudioEngineImpl::findSound(const std::string &path)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
std::string fullPath = FileUtils::getInstance()->fullPathForFilename(path);
|
|
|
|
std::map<std::string, FMOD::Sound *>::const_iterator it = mapSound.find(fullPath);
|
2016-05-19 04:17:33 +08:00
|
|
|
return (it != mapSound.end()) ? (it->second) : nullptr;
|
2015-11-25 09:54:12 +08:00
|
|
|
}
|
|
|
|
|
2016-05-19 04:17:33 +08:00
|
|
|
FMOD::Channel * AudioEngineImpl::getChannel(FMOD::Sound *sound)
|
|
|
|
{
|
2016-05-19 02:34:04 +08:00
|
|
|
void * data;
|
|
|
|
sound->getUserData(&data);
|
2016-05-20 00:39:40 +08:00
|
|
|
int id = static_cast<int>(reinterpret_cast<std::intptr_t>(data));
|
2016-05-19 02:34:04 +08:00
|
|
|
return mapChannelInfo[id].channel;
|
2016-05-19 00:40:00 +08:00
|
|
|
}
|