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

623 lines
16 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 "config.h"
#include "player.h"
#include "engines/enginebase.h"
#include "playlist/playlist.h"
#include "playlist/playlistitem.h"
#include "playlist/playlistmanager.h"
#include "radio/lastfmservice.h"
#ifdef HAVE_GSTREAMER
# include "engines/gstengine.h"
#endif
#ifdef HAVE_LIBVLC
# include "engines/vlcengine.h"
#endif
#ifdef HAVE_LIBXINE
# include "engines/xine-engine.h"
#endif
#ifdef HAVE_QT_PHONON
# include "engines/phononengine.h"
#endif
#ifdef Q_WS_X11
# include "core/mpris_player.h"
# include "core/mpris_tracklist.h"
# include <QDBusConnection>
#endif
#include <QtDebug>
#include <QtConcurrentRun>
#include <boost/bind.hpp>
using boost::shared_ptr;
const char* Player::kRainUrl = "http://data.clementine-player.org/rainymood";
const char* Player::kHypnotoadUrl = "http://data.clementine-player.org/hypnotoad";
#ifdef Q_WS_X11
QDBusArgument& operator<< (QDBusArgument& arg, const DBusStatus& status) {
arg.beginStructure();
arg << status.play;
arg << status.random;
arg << status.repeat;
arg << status.repeat_playlist;
arg.endStructure();
return arg;
}
const QDBusArgument& operator>> (const QDBusArgument& arg, DBusStatus& status) {
arg.beginStructure();
arg >> status.play;
arg >> status.random;
arg >> status.repeat;
arg >> status.repeat_playlist;
arg.endStructure();
return arg;
}
#endif
Player::Player(PlaylistManager* playlists, LastFMService* lastfm,
Engine::Type engine, QObject* parent)
: QObject(parent),
playlists_(playlists),
lastfm_(lastfm),
engine_(CreateEngine(engine)),
stream_change_type_(Engine::First),
rain_stream_(-1),
toad_stream_(-1)
{
settings_.beginGroup("Player");
SetVolume(settings_.value("volume", 50).toInt());
connect(engine_.get(), SIGNAL(Error(QString)), SIGNAL(Error(QString)));
// MPRIS DBus interface.
#ifdef Q_WS_X11
MprisPlayer* mpris = new MprisPlayer(this);
// Hack so the next registerObject() doesn't override this one.
QDBusConnection::sessionBus().registerObject(
"/Player", mpris, QDBusConnection::ExportAllSlots | QDBusConnection::ExportAllSignals);
new MprisTrackList(this);
QDBusConnection::sessionBus().registerObject("/TrackList", this);
#endif
}
Player::~Player() {
}
EngineBase* Player::CreateEngine(Engine::Type engine) {
switch(engine) {
#ifdef HAVE_GSTREAMER
case Engine::Type_GStreamer:
return new GstEngine();
break;
#endif
#ifdef HAVE_LIBVLC
case Engine::Type_VLC:
return new VlcEngine();
break;
#endif
#ifdef HAVE_LIBXINE
case Engine::Type_Xine:
return new XineEngine();
break;
#endif
#ifdef HAVE_QT_PHONON
case Engine::Type_QtPhonon:
return new PhononEngine();
break;
#endif
default:
qFatal("Selected engine not compiled in");
break;
}
/* NOT REACHED */
return NULL;
}
void Player::Init() {
if (!engine_->Init())
qFatal("Error initialising audio engine");
connect(engine_.get(), SIGNAL(StateChanged(Engine::State)), SLOT(EngineStateChanged(Engine::State)));
connect(engine_.get(), SIGNAL(TrackAboutToEnd()), SLOT(TrackAboutToEnd()));
connect(engine_.get(), SIGNAL(TrackEnded()), SLOT(TrackEnded()));
connect(engine_.get(), SIGNAL(MetaData(Engine::SimpleMetaBundle)),
SLOT(EngineMetadataReceived(Engine::SimpleMetaBundle)));
engine_->SetVolume(settings_.value("volume", 50).toInt());
}
void Player::ReloadSettings() {
engine_->ReloadSettings();
}
void Player::HandleSpecialLoad(const PlaylistItem::SpecialLoadResult &result) {
switch (result.type_) {
case PlaylistItem::SpecialLoadResult::NoMoreTracks:
loading_async_ = QUrl();
NextItem(Engine::Auto);
break;
case PlaylistItem::SpecialLoadResult::TrackAvailable: {
// Might've been an async load, so check we're still on the same item
int current_index = playlists_->active()->current_index();
if (current_index == -1)
return;
shared_ptr<PlaylistItem> item = playlists_->active()->item_at(current_index);
if (!item || item->Url() != result.original_url_)
return;
engine_->Play(result.media_url_, stream_change_type_);
current_item_ = item;
loading_async_ = QUrl();
break;
}
case PlaylistItem::SpecialLoadResult::WillLoadAsynchronously:
// We'll get called again later with either NoMoreTracks or TrackAvailable
loading_async_ = result.original_url_;
break;
}
}
void Player::Next() {
NextInternal(Engine::Manual);
}
void Player::NextInternal(Engine::TrackChangeType change) {
if (playlists_->active()->current_item() &&
playlists_->active()->current_item()->options() & PlaylistItem::ContainsMultipleTracks) {
// The next track is already being loaded
if (playlists_->active()->current_item()->Url() == loading_async_)
return;
stream_change_type_ = change;
HandleSpecialLoad(playlists_->active()->current_item()->LoadNext());
return;
}
NextItem(change);
}
void Player::NextItem(Engine::TrackChangeType change) {
int i = playlists_->active()->next_index();
playlists_->active()->set_current_index(i);
if (i == -1) {
emit PlaylistFinished();
Stop();
return;
}
PlayAt(i, change, false);
}
void Player::TrackEnded() {
if (playlists_->active()->stop_after_current()) {
Stop();
return;
}
NextInternal(Engine::Auto);
}
void Player::PlayPause() {
switch (engine_->state()) {
case Engine::Paused:
engine_->Unpause();
break;
case Engine::Playing: {
// We really shouldn't pause last.fm streams
// Stopping seems like a reasonable thing to do (especially on mac where there
// is no media key for stop).
if (current_item_->options() & PlaylistItem::PauseDisabled) {
engine_->Stop();
} else {
engine_->Pause();
}
break;
}
case Engine::Empty:
case Engine::Idle: {
playlists_->SetActivePlaylist(playlists_->current_id());
if (playlists_->active()->rowCount() == 0)
break;
int i = playlists_->active()->current_index();
if (i == -1) i = playlists_->active()->last_played_index();
if (i == -1) i = 0;
PlayAt(i, Engine::First, true);
break;
}
}
}
void Player::Stop() {
engine_->Stop();
playlists_->active()->set_current_index(-1);
current_item_.reset();
}
void Player::Previous() {
int i = playlists_->active()->previous_index();
playlists_->active()->set_current_index(i);
if (i == -1) {
Stop();
return;
}
PlayAt(i, Engine::Manual, false);
}
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;
}
emit StatusChange(GetStatus());
emit CapsChange(GetCaps());
}
void Player::SetVolume(int value) {
int old_volume = engine_->volume();
int volume = qBound(0, value, 100);
settings_.setValue("volume", volume);
engine_->SetVolume(volume);
if (volume != old_volume)
emit VolumeChanged(volume);
}
int Player::GetVolume() const {
return engine_->volume();
}
Engine::State Player::GetState() const {
return engine_->state();
}
void Player::PlayAt(int index, Engine::TrackChangeType change, bool reshuffle) {
if (reshuffle)
playlists_->active()->set_current_index(-1);
playlists_->active()->set_current_index(index);
current_item_ = playlists_->active()->item_at(index);
if (current_item_->options() & PlaylistItem::SpecialPlayBehaviour) {
// It's already loading
if (current_item_->Url() == loading_async_)
return;
stream_change_type_ = change;
HandleSpecialLoad(current_item_->StartLoading());
}
else {
loading_async_ = QUrl();
engine_->Play(current_item_->Url(), change);
if (lastfm_->IsScrobblingEnabled())
lastfm_->NowPlaying(current_item_->Metadata());
}
emit CapsChange(GetCaps());
}
void Player::CurrentMetadataChanged(const Song &metadata) {
lastfm_->NowPlaying(metadata);
emit TrackChange(GetMetadata());
}
void Player::Seek(int seconds) {
int msec = qBound(0, seconds * 1000, int(engine_->length()));
engine_->Seek(msec);
// If we seek the track we don't want to submit it to last.fm
playlists_->active()->set_scrobbled(true);
}
void Player::EngineMetadataReceived(const Engine::SimpleMetaBundle& bundle) {
shared_ptr<PlaylistItem> item = playlists_->active()->current_item();
if (item == NULL)
return;
Engine::SimpleMetaBundle bundle_copy = bundle;
// Maybe the metadata is from icycast and has "Artist - Title" shoved
// together in the title field.
int dash_pos = bundle_copy.title.indexOf('-');
if (dash_pos != -1 && bundle_copy.artist.isEmpty()) {
bundle_copy.artist = bundle_copy.title.mid(dash_pos + 1).trimmed();
bundle_copy.title = bundle_copy.title.left(dash_pos).trimmed();
}
Song song = item->Metadata();
song.MergeFromSimpleMetaBundle(bundle_copy);
// Ignore useless metadata
if (song.title().isEmpty() && song.artist().isEmpty())
return;
playlists_->active()->SetStreamMetadata(item->Url(), song);
}
int Player::GetCaps() const {
int caps = CAN_HAS_TRACKLIST;
if (current_item_) { caps |= CAN_PROVIDE_METADATA; }
if (GetState() == Engine::Playing && current_item_->options() & PlaylistItem::PauseDisabled) {
caps |= CAN_PAUSE;
}
if (GetState() == Engine::Paused) {
caps |= CAN_PLAY;
}
if (GetState() != Engine::Empty && current_item_->Metadata().filetype() != Song::Type_Stream) {
caps |= CAN_SEEK;
}
if (playlists_->active()->next_index() != -1 ||
playlists_->active()->current_item_options() & PlaylistItem::ContainsMultipleTracks) {
caps |= CAN_GO_NEXT;
}
if (playlists_->active()->previous_index() != -1) {
caps |= CAN_GO_PREV;
}
return caps;
}
DBusStatus Player::GetStatus() const {
DBusStatus status;
switch (GetState()) {
case Engine::Empty:
case Engine::Idle:
status.play = DBusStatus::Mpris_Stopped;
break;
case Engine::Playing:
status.play = DBusStatus::Mpris_Playing;
break;
case Engine::Paused:
status.play = DBusStatus::Mpris_Paused;
break;
}
status.random = playlists_->sequence()->shuffle_mode() == PlaylistSequence::Shuffle_Off ? 0 : 1;
PlaylistSequence::RepeatMode repeat_mode = playlists_->sequence()->repeat_mode();
status.repeat = repeat_mode == PlaylistSequence::Repeat_Track ? 1 : 0;
status.repeat_playlist = (repeat_mode == PlaylistSequence::Repeat_Album ||
repeat_mode == PlaylistSequence::Repeat_Playlist) ? 1 : 0;
return status;
}
namespace {
inline void AddMetadata(const QString& key, const QString& metadata, QVariantMap* map) {
if (!metadata.isEmpty()) {
(*map)[key] = metadata;
}
}
inline void AddMetadata(const QString& key, int metadata, QVariantMap* map) {
if (metadata > 0) {
(*map)[key] = metadata;
}
}
} // namespace
QVariantMap Player::GetMetadata(const PlaylistItem& item) const {
QVariantMap ret;
const Song& song = item.Metadata();
AddMetadata("location", item.Url().toString(), &ret);
AddMetadata("title", song.PrettyTitle(), &ret);
AddMetadata("artist", song.artist(), &ret);
AddMetadata("album", song.album(), &ret);
AddMetadata("time", song.length(), &ret);
AddMetadata("tracknumber", song.track(), &ret);
return ret;
}
QVariantMap Player::GetMetadata() const {
shared_ptr<PlaylistItem> item = playlists_->active()->current_item();
if (item) {
return GetMetadata(*item);
}
return QVariantMap();
}
QVariantMap Player::GetMetadata(int track) const {
if (track >= playlists_->active()->rowCount() || track < 0) {
return QVariantMap();
}
const PlaylistItem& item = *(playlists_->active()->item_at(track));
return GetMetadata(item);
}
void Player::Mute() {
SetVolume(0);
}
void Player::Pause() {
switch (GetState()) {
case Engine::Playing:
engine_->Pause();
break;
case Engine::Paused:
engine_->Unpause();
break;
default:
return;
}
}
void Player::Play() {
switch (GetState()) {
case Engine::Playing:
Seek(0);
break;
case Engine::Paused:
engine_->Unpause();
break;
default:
PlayPause();
break;
}
}
void Player::Prev() {
Previous();
}
int Player::PositionGet() const {
return engine_->position();
}
void Player::PositionSet(int x) {
Seek(x / 1000);
}
void Player::Repeat(bool enable) {
playlists_->sequence()->SetRepeatMode(
enable ? PlaylistSequence::Repeat_Track : PlaylistSequence::Repeat_Off);
}
void Player::ShowOSD() {
if (current_item_)
emit ForceShowOSD(current_item_->Metadata());
}
void Player::VolumeDown(int change) {
SetVolume(GetVolume() - change);
}
void Player::VolumeUp(int change) {
SetVolume(GetVolume() + change);
}
int Player::VolumeGet() const {
return GetVolume();
}
void Player::VolumeSet(int volume) {
SetVolume(volume);
}
int Player::AddTrack(const QString& track, bool play_now) {
playlists_->active()->InsertUrls(QList<QUrl>() << QUrl(track), play_now);
return 0;
}
void Player::DelTrack(int index) {
playlists_->active()->removeRows(index, 1);
}
int Player::GetCurrentTrack() const {
return playlists_->active()->current_index();
}
int Player::GetLength() const {
return playlists_->active()->rowCount();
}
void Player::SetLoop(bool enable) {
playlists_->active()->sequence()->SetRepeatMode(
enable ? PlaylistSequence::Repeat_Playlist : PlaylistSequence::Repeat_Off);
}
void Player::SetRandom(bool enable) {
playlists_->active()->sequence()->SetShuffleMode(
enable ? PlaylistSequence::Shuffle_All : PlaylistSequence::Shuffle_Off);
}
void Player::PlayTrack(int index) {
PlayAt(index, Engine::Manual, true);
}
void Player::PlaylistChanged() {
emit TrackListChange(GetLength());
}
void Player::TrackAboutToEnd() {
if (engine_->is_autocrossfade_enabled()) {
// Crossfade is on, so just start playing the next track. The current one
// will fade out, and the new one will fade in
NextInternal(Engine::Auto);
} else {
// Crossfade is off, so start preloading the next track so we don't get a
// gap between songs.
if (current_item_->options() & PlaylistItem::ContainsMultipleTracks)
return;
if (playlists_->active()->next_index() == -1)
return;
shared_ptr<PlaylistItem> item = playlists_->active()->item_at(
playlists_->active()->next_index());
if (!item)
return;
QUrl url = item->Url();
// Get the actual track URL rather than the stream URL.
if (item->options() & PlaylistItem::ContainsMultipleTracks) {
PlaylistItem::SpecialLoadResult result = item->LoadNext();
switch (result.type_) {
case PlaylistItem::SpecialLoadResult::NoMoreTracks:
return;
case PlaylistItem::SpecialLoadResult::WillLoadAsynchronously:
loading_async_ = item->Url();
return;
case PlaylistItem::SpecialLoadResult::TrackAvailable:
url = result.media_url_;
break;
}
}
engine_->StartPreloading(url);
}
}
void Player::MakeItRain(bool rain) {
const bool is_raining = rain_stream_ != -1;
if (rain && !is_raining) {
rain_stream_ = engine_->AddBackgroundStream(QUrl(kRainUrl));
}
if (!rain && is_raining) {
engine_->StopBackgroundStream(rain_stream_);
rain_stream_ = -1;
}
}
void Player::AllHail(bool hypnotoad) {
const bool is_hailing = toad_stream_ != -1;
if (hypnotoad && !is_hailing) {
toad_stream_ = engine_->AddBackgroundStream(QUrl(kHypnotoadUrl));
}
if (!hypnotoad && is_hailing) {
engine_->StopBackgroundStream(toad_stream_);
toad_stream_ = -1;
}
}