Clementine-audio-player-Mac.../src/player.cpp

251 lines
5.6 KiB
C++

/* This file is part of Clementine.
Clementine is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Clementine 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with Clementine. If not, see <http://www.gnu.org/licenses/>.
*/
#include "player.h"
#include "playlist.h"
#include "lastfmservice.h"
#ifdef Q_OS_WIN32
# include "phononengine.h"
#else
# include "xine-engine.h"
#endif
#include <QtDebug>
#include <QtConcurrentRun>
#include <boost/bind.hpp>
Player::Player(Playlist* playlist, LastFMService* lastfm, QObject* parent)
: QObject(parent),
playlist_(playlist),
lastfm_(lastfm),
current_item_options_(PlaylistItem::Default),
engine_(NULL),
init_engine_watcher_(new QFutureWatcher<bool>(this))
{
#ifdef Q_OS_WIN32
engine_ = new PhononEngine;
#else
engine_ = new XineEngine;
#endif
settings_.beginGroup("Player");
SetVolume(settings_.value("volume", 50).toInt());
connect(init_engine_watcher_, SIGNAL(finished()), SLOT(EngineInitFinished()));
connect(engine_, SIGNAL(error(QString)), SIGNAL(Error(QString)));
}
void Player::Init() {
init_engine_ = QtConcurrent::run(boost::bind(&EngineBase::init, engine_));
init_engine_watcher_->setFuture(init_engine_);
}
void Player::EngineInitFinished() {
if (init_engine_.result() == false) {
qFatal("Error initialising audio engine");
}
connect(engine_, SIGNAL(stateChanged(Engine::State)), SLOT(EngineStateChanged(Engine::State)));
connect(engine_, SIGNAL(trackEnded()), SLOT(TrackEnded()));
connect(engine_, SIGNAL(metaData(Engine::SimpleMetaBundle)),
SLOT(EngineMetadataReceived(Engine::SimpleMetaBundle)));
engine_->setVolume(settings_.value("volume", 50).toInt());
emit InitFinished();
}
void Player::ReloadSettings() {
if (!init_engine_.isFinished())
return;
engine_->reloadSettings();
}
void Player::Next() {
if (playlist_->current_item_options() & PlaylistItem::ContainsMultipleTracks) {
playlist_->current_item()->LoadNext();
return;
}
NextItem();
}
void Player::NextItem() {
int i = playlist_->next_index();
playlist_->set_current_index(i);
if (i == -1) {
Stop();
return;
}
PlayAt(i);
}
void Player::TrackEnded() {
int i = playlist_->current_index();
if (i == -1 || playlist_->stop_after_current()) {
Stop();
return;
}
Next();
}
void Player::PlayPause() {
if (!init_engine_.isFinished())
return;
switch (engine_->state()) {
case Engine::Paused:
qDebug() << "Unpausing";
engine_->unpause();
break;
case Engine::Playing:
// We really shouldn't pause last.fm streams
if (current_item_options_ & PlaylistItem::PauseDisabled)
break;
qDebug() << "Pausing";
engine_->pause();
break;
case Engine::Empty:
case Engine::Idle: {
int i = playlist_->current_index();
if (i == -1) {
if (playlist_->rowCount() == 0)
break;
i = 0;
}
PlayAt(i);
break;
}
}
}
void Player::Stop() {
if (!init_engine_.isFinished())
return;
qDebug() << "Stopping";
engine_->stop();
playlist_->set_current_index(-1);
}
void Player::Previous() {
int i = playlist_->previous_index();
playlist_->set_current_index(i);
if (i == -1) {
Stop();
return;
}
PlayAt(i);
}
void Player::EngineStateChanged(Engine::State state) {
switch (state) {
case Engine::Paused: emit Paused(); break;
case Engine::Playing: emit Playing(); break;
case Engine::Empty:
case Engine::Idle: emit Stopped(); break;
}
}
void Player::SetVolume(int value) {
settings_.setValue("volume", value);
engine_->setVolume(value);
emit VolumeChanged(value);
}
int Player::GetVolume() const {
return engine_->volume();
}
Engine::State Player::GetState() const {
return engine_->state();
}
void Player::PlayAt(int index) {
if (!init_engine_.isFinished())
return;
playlist_->set_current_index(index);
PlaylistItem* item = playlist_->item_at(index);
current_item_options_ = item->options();
current_item_ = item->Metadata();
if (item->options() & PlaylistItem::SpecialPlayBehaviour)
item->StartLoading();
else {
engine_->play(item->Url());
if (lastfm_->IsScrobblingEnabled())
lastfm_->NowPlaying(item->Metadata());
}
}
void Player::StreamReady(const QUrl& original_url, const QUrl& media_url) {
if (!init_engine_.isFinished())
return;
int current_index = playlist_->current_index();
if (current_index == -1)
return;
PlaylistItem* item = playlist_->item_at(current_index);
if (!item || item->Url() != original_url)
return;
engine_->play(media_url);
current_item_ = item->Metadata();
current_item_options_ = item->options();
}
void Player::CurrentMetadataChanged(const Song &metadata) {
lastfm_->NowPlaying(metadata);
current_item_ = metadata;
}
void Player::Seek(int seconds) {
if (!init_engine_.isFinished())
return;
engine_->seek(seconds * 1000);
}
void Player::EngineMetadataReceived(const Engine::SimpleMetaBundle& bundle) {
PlaylistItem* item = playlist_->current_item();
if (item == NULL)
return;
Song song;
song.InitFromSimpleMetaBundle(bundle);
// Ignore useless metadata
if (song.title().isEmpty() && song.artist().isEmpty())
return;
playlist_->SetStreamMetadata(item->Url(), song);
}