/* This file is part of the KDE project. Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). This library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2.1 or 3 of the License. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library. If not, see . */ #include #include "audioplayer.h" #include "utils.h" QT_BEGIN_NAMESPACE using namespace Phonon; using namespace Phonon::MMF; /*! \class MMF::AudioPlayer \internal */ //----------------------------------------------------------------------------- // Constructor / destructor //----------------------------------------------------------------------------- MMF::AudioPlayer::AudioPlayer() { construct(); } MMF::AudioPlayer::AudioPlayer(const AbstractPlayer& player) : AbstractMediaPlayer(player) { construct(); } void MMF::AudioPlayer::construct() { TRACE_CONTEXT(AudioPlayer::AudioPlayer, EAudioApi); TRACE_ENTRY_0(); CPlayerType *player = 0; QT_TRAP_THROWING(player = CPlayerType::NewL(*this, 0, EMdaPriorityPreferenceNone)); m_player.reset(player); m_player->RegisterForAudioLoadingNotification(*this); TRACE_EXIT_0(); } MMF::AudioPlayer::~AudioPlayer() { TRACE_CONTEXT(AudioPlayer::~AudioPlayer, EAudioApi); TRACE_ENTRY_0(); TRACE_EXIT_0(); } //----------------------------------------------------------------------------- // Public API //----------------------------------------------------------------------------- void MMF::AudioPlayer::doPlay() { m_player->Play(); } void MMF::AudioPlayer::doPause() { m_player->Pause(); } void MMF::AudioPlayer::doStop() { m_player->Stop(); } void MMF::AudioPlayer::doSeek(qint64 ms) { m_player->SetPosition(TTimeIntervalMicroSeconds(ms * 1000)); } int MMF::AudioPlayer::setDeviceVolume(int mmfVolume) { /* In SDK 3.1, SetVolume() returns void. If we're compiling against * 3.1, we handle it with ifdefs. However, if we compile against a later * SDK but are _running_ against 3.1, we avoid returning from an undefined * stack by doing a runtime check of the SDK version. */ #if !defined(__SERIES60_31__) const int err = m_player->SetVolume(mmfVolume); if (QSysInfo::s60Version() >= QSysInfo::SV_S60_5_0) return err; else return KErrNone; #else m_player->SetVolume(mmfVolume); return KErrNone; #endif } int MMF::AudioPlayer::openFile(RFile& file) { TRAPD(err, m_player->OpenFileL(file)); #ifdef QT_PHONON_MMF_AUDIO_DRM if (KErrNone == err) { // There appears to be a bug in the CDrmPlayerUtility implementation (at least // in S60 5.x) whereby the player does not check whether the loading observer // pointer is null before dereferencing it. Therefore we must register for // loading notification, even though we do nothing in the callback functions. m_player->RegisterForAudioLoadingNotification(*this); } #endif return err; } int MMF::AudioPlayer::openUrl(const QString& /*url*/) { // Streaming playback is generally not supported by the implementation // of the audio player API, so we use CVideoPlayerUtility for both // audio and video streaming. Utils::panic(AudioUtilityUrlNotSupported); // Silence warning return 0; } int MMF::AudioPlayer::bufferStatus() const { int result = 0; TRAP_IGNORE(m_player->GetAudioLoadingProgressL(result)); return result; } void MMF::AudioPlayer::close() { m_player->Close(); } bool MMF::AudioPlayer::hasVideo() const { return false; } qint64 MMF::AudioPlayer::currentTime() const { TRACE_CONTEXT(AudioPlayer::currentTime, EAudioApi); TTimeIntervalMicroSeconds us; const TInt err = m_player->GetPosition(us); qint64 result = 0; if (KErrNone == err) { result = toMilliSeconds(us); } else { TRACE("GetPosition err %d", err); // If we don't cast away constness here, we simply have to ignore // the error. const_cast(this)->setError(tr("Getting position failed"), err); } return result; } qint64 MMF::AudioPlayer::totalTime() const { return toMilliSeconds(m_player->Duration()); } //----------------------------------------------------------------------------- // Symbian multimedia client observer callbacks //----------------------------------------------------------------------------- #ifdef QT_PHONON_MMF_AUDIO_DRM void MMF::AudioPlayer::MdapcInitComplete(TInt aError, const TTimeIntervalMicroSeconds &) #else void MMF::AudioPlayer::MapcInitComplete(TInt aError, const TTimeIntervalMicroSeconds &) #endif { TRACE_CONTEXT(AudioPlayer::MapcInitComplete, EAudioInternal); TRACE_ENTRY("state %d error %d", state(), aError); __ASSERT_ALWAYS(LoadingState == state(), Utils::panic(InvalidStatePanic)); if (KErrNone == aError) { maxVolumeChanged(m_player->MaxVolume()); emit totalTimeChanged(totalTime()); updateMetaData(); changeState(StoppedState); } else { setError(tr("Opening clip failed"), aError); } TRACE_EXIT_0(); } #ifdef QT_PHONON_MMF_AUDIO_DRM void MMF::AudioPlayer::MdapcPlayComplete(TInt aError) #else void MMF::AudioPlayer::MapcPlayComplete(TInt aError) #endif { TRACE_CONTEXT(AudioPlayer::MapcPlayComplete, EAudioInternal); TRACE_ENTRY("state %d error %d", state(), aError); if (KErrNone == aError) { changeState(StoppedState); // TODO: move on to m_nextSource } else { setError(tr("Playback complete"), aError); } /* if (aError == KErrNone) { if (m_nextSource.type() == MediaSource::Empty) { emit finished(); } else { setSource(m_nextSource); m_nextSource = MediaSource(); } changeState(StoppedState); } else { m_error = NormalError; changeState(ErrorState); } */ TRACE_EXIT_0(); } CPlayerType *MMF::AudioPlayer::player() const { return m_player.data(); } #ifdef QT_PHONON_MMF_AUDIO_DRM void MMF::AudioPlayer::MaloLoadingStarted() { } void MMF::AudioPlayer::MaloLoadingComplete() { } #endif // QT_PHONON_MMF_AUDIO_DRM //----------------------------------------------------------------------------- // MAudioLoadingObserver callbacks //----------------------------------------------------------------------------- void MMF::AudioPlayer::MaloLoadingStarted() { bufferingStarted(); } void MMF::AudioPlayer::MaloLoadingComplete() { bufferingComplete(); } //----------------------------------------------------------------------------- // Private functions //----------------------------------------------------------------------------- int MMF::AudioPlayer::numberOfMetaDataEntries() const { int numberOfEntries = 0; m_player->GetNumberOfMetaDataEntries(numberOfEntries); // ignoring return code return numberOfEntries; } QPair MMF::AudioPlayer::metaDataEntry(int index) const { CMMFMetaDataEntry *entry = 0; QT_TRAP_THROWING(entry = m_player->GetMetaDataEntryL(index)); return QPair(qt_TDesC2QString(entry->Name()), qt_TDesC2QString(entry->Value())); } QT_END_NAMESPACE