/*************************************************************************** * Copyright (C) 2010-2013 by Ilya Kotov * * forkotov02@hotmail.ru * * * * This program 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 2 of the License, or * * (at your option) any later version. * * * * This program 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 this program; if not, write to the * * Free Software Foundation, Inc., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include "player2object.h" Player2Object::Player2Object(QObject *parent) : QDBusAbstractAdaptor(parent) { m_prev_track = 0; m_previous_pos = 0; m_core = SoundCore::instance(); m_player = MediaPlayer::instance(); m_pl_manager = m_player->playListManager(); m_ui_settings = QmmpUiSettings::instance(); connect(m_core, SIGNAL(metaDataChanged ()), SLOT(updateId())); connect(m_core, SIGNAL(metaDataChanged ()), SLOT(emitPropertiesChanged())); connect(m_core, SIGNAL(stateChanged (Qmmp::State)), SLOT(checkState(Qmmp::State))); connect(m_core, SIGNAL(stateChanged (Qmmp::State)), SLOT(emitPropertiesChanged())); connect(m_core, SIGNAL(volumeChanged(int,int)), SLOT(emitPropertiesChanged())); connect(m_core, SIGNAL(elapsedChanged(qint64)), SLOT(checkSeeking(qint64))); connect(m_ui_settings, SIGNAL(repeatableListChanged(bool)), SLOT(emitPropertiesChanged())); connect(m_ui_settings, SIGNAL(shuffleChanged(bool)), SLOT(emitPropertiesChanged())); connect(m_pl_manager, SIGNAL(currentPlayListChanged(PlayListModel*,PlayListModel*)), SLOT(setModel(PlayListModel*,PlayListModel*))); connect(m_ui_settings, SIGNAL(repeatableListChanged(bool)), SLOT(emitPropertiesChanged())); setModel(m_pl_manager->currentPlayList(), 0); updateId(); syncProperties(); } Player2Object::~Player2Object() {} bool Player2Object::canControl() const { return true; } bool Player2Object::canGoNext() const { return m_pl_manager->currentPlayList()->nextTrack() != 0; } bool Player2Object::canGoPrevious() const { return m_pl_manager->currentPlayList()->currentIndex() > 0; } bool Player2Object::canPause() const { return (m_core->state() == Qmmp::Paused || m_core->state() == Qmmp::Playing); } bool Player2Object::canPlay() const { return m_pl_manager->currentPlayList()->count() != 0; } bool Player2Object::canSeek() const { return m_core->totalTime() > 0; } QString Player2Object::loopStatus() const { if(m_ui_settings->isRepeatableTrack()) return "Track"; else if(m_ui_settings->isRepeatableList()) return "Playlist"; else return "None"; } void Player2Object::setLoopStatus(const QString &value) { if(value == "Track") { m_ui_settings->setRepeatableList(false); m_ui_settings->setRepeatableTrack(true); } else if(value == "Playlist") { m_ui_settings->setRepeatableList(true); m_ui_settings->setRepeatableTrack(false); } else { m_ui_settings->setRepeatableList(false); m_ui_settings->setRepeatableTrack(false); } } double Player2Object::maximumRate() const { return 1.0; } QVariantMap Player2Object::metadata() const { PlayListTrack *track = m_pl_manager->currentPlayList()->currentTrack(); if(!track || m_core->metaData(Qmmp::URL).isEmpty()) return QVariantMap(); QVariantMap map; map["mpris:length"] = qMax(m_core->totalTime() * 1000 , qint64(0)); if(!MetaDataManager::instance()->getCoverPath(m_core->metaData(Qmmp::URL)).isEmpty()) { map["mpris:artUrl"] = QUrl::fromLocalFile( MetaDataManager::instance()->getCoverPath(m_core->metaData(Qmmp::URL))).toString(); } if(!m_core->metaData(Qmmp::ALBUM).isEmpty()) map["xesam:album"] = m_core->metaData(Qmmp::ALBUM); if(!m_core->metaData(Qmmp::ARTIST).isEmpty()) map["xesam:artist"] = QStringList() << m_core->metaData(Qmmp::ARTIST); if(!m_core->metaData(Qmmp::COMMENT).isEmpty()) map["xesam:comment"] = QStringList() << m_core->metaData(Qmmp::COMMENT); if(!m_core->metaData(Qmmp::COMPOSER).isEmpty()) map["xesam:composer"] = QStringList() << m_core->metaData(Qmmp::COMPOSER); if(!m_core->metaData(Qmmp::DISCNUMBER).isEmpty()) map["xesam:discNumber"] = m_core->metaData(Qmmp::DISCNUMBER).toInt(); if(!m_core->metaData(Qmmp::GENRE).isEmpty()) map["xesam:genre"] = QStringList() << m_core->metaData(Qmmp::GENRE); if(!m_core->metaData(Qmmp::TITLE).isEmpty()) map["xesam:title"] = m_core->metaData(Qmmp::TITLE); if(!m_core->metaData(Qmmp::TRACK).isEmpty()) map["xesam:trackNumber"] = m_core->metaData(Qmmp::TRACK).toInt(); map["mpris:trackid"] = QVariant::fromValue(m_trackID); if(m_core->metaData(Qmmp::URL).startsWith("/")) map["xesam:url"] = QUrl::fromLocalFile(m_core->metaData(Qmmp::URL)).toString(); else map["xesam:url"] = m_core->metaData(Qmmp::URL); return map; } double Player2Object::minimumRate() const { return 1.0; } QString Player2Object::playbackStatus() const { if(m_core->state() == Qmmp::Playing) return "Playing"; else if (m_core->state() == Qmmp::Paused) return "Paused"; return "Stopped"; } qlonglong Player2Object::position() const { return qMax(m_core->elapsed() * 1000, qint64(0)); } double Player2Object::rate() const { return 1.0; } void Player2Object::setRate(double value) { Q_UNUSED(value) } bool Player2Object::shuffle() const { return m_ui_settings->isShuffle(); } void Player2Object::setShuffle(bool value) { m_ui_settings->setShuffle(value); } double Player2Object::volume() const { return qMax(m_core->leftVolume(), m_core->rightVolume())/100.0; } void Player2Object::Player2Object::setVolume(double value) { value = qBound(0.0, value ,1.0); int balance = (volume() > 0) ? (m_core->rightVolume() - m_core->leftVolume())/volume() : 0; m_core->setVolume(value*100 - qMax(balance,0)*value, value*100 + qMin(balance,0)*value); } void Player2Object::Next() { m_player->next(); } void Player2Object::OpenUri(const QString &in0) { QString path = in0; if(in0.startsWith("file://")) { path = QUrl(in0).toLocalFile (); if(!QFile::exists(path)) return; //error } if(!m_pl_manager->currentPlayList()->isLoaderRunning()) { m_pl_manager->selectPlayList(m_pl_manager->currentPlayList()); connect(m_pl_manager->currentPlayList(), SIGNAL(trackAdded(PlayListTrack*)), SLOT(playTrack(PlayListTrack*))); connect(m_pl_manager->currentPlayList(), SIGNAL(loaderFinished()), this, SLOT(disconnectPl())); } m_pl_manager->currentPlayList()->add(path); } void Player2Object::Pause() { m_core->pause(); } void Player2Object::Play() { m_player->play(); } void Player2Object::PlayPause() { if(m_core->state() == Qmmp::Stopped) m_player->play(); else if(m_core->state() == Qmmp::Paused || m_core->state() == Qmmp::Playing) m_core->pause(); } void Player2Object::Previous() { m_player->previous(); } void Player2Object::Seek(qlonglong Offset) { m_core->seek(qMin((qint64)0, m_core->elapsed() + Offset/1000)); } void Player2Object::SetPosition(const QDBusObjectPath &TrackId, qlonglong Position) { if(m_trackID == TrackId) m_core->seek(Position/1000); else qWarning("Player2Object: SetPosition() called with a invalid trackId"); } void Player2Object::Stop() { m_core->stop(); } void Player2Object::emitPropertiesChanged() { QList changedProps; if(m_props["CanGoNext"] != canGoNext()) changedProps << "CanGoNext"; if(m_props["CanGoPrevious"] != canGoPrevious()) changedProps << "CanGoPrevious"; if(m_props["CanPause"] != canPause()) changedProps << "CanPause"; if(m_props["CanPlay"] != canPlay()) changedProps << "CanPlay"; if(m_props["CanSeek"] != canSeek()) changedProps << "CanSeek"; if(m_props["LoopStatus"] != loopStatus()) changedProps << "LoopStatus"; if(m_props["MaximumRate"] != maximumRate()) changedProps << "MaximumRate"; if(m_props["MinimumRate"] != minimumRate()) changedProps << "MinimumRate"; if(m_props["PlaybackStatus"] != playbackStatus()) changedProps << "PlaybackStatus"; if(m_props["Rate"] != rate()) changedProps << "Rate"; if(m_props["Shuffle"] != shuffle()) changedProps << "Shuffle"; if(m_props["Volume"] != volume()) changedProps << "Volume"; if(m_props["Metadata"] != metadata()) changedProps << "Metadata"; if(changedProps.isEmpty()) return; syncProperties(); QVariantMap map; foreach(QByteArray name, changedProps) map.insert(name, m_props.value(name)); QDBusMessage msg = QDBusMessage::createSignal("/org/mpris/MediaPlayer2", "org.freedesktop.DBus.Properties", "PropertiesChanged"); msg << "org.mpris.MediaPlayer2.Player"; msg << map; msg << QStringList(); QDBusConnection::sessionBus().send(msg); } void Player2Object::updateId() { if(m_prev_track != m_pl_manager->currentPlayList()->currentTrack()) { m_trackID = QDBusObjectPath(QString("%1/Track/%2").arg("/org/qmmp/MediaPlayer2").arg(qrand())); m_prev_track = m_pl_manager->currentPlayList()->currentTrack(); } } void Player2Object::checkState(Qmmp::State state) { if(state == Qmmp::Playing) { updateId(); m_previous_pos = 0; } } void Player2Object::checkSeeking(qint64 elapsed) { if(abs(elapsed - m_previous_pos) > 2000) { emit Seeked(elapsed * 1000); } m_previous_pos = elapsed; } void Player2Object::playTrack(PlayListTrack *item) { m_pl_manager->selectPlayList((PlayListModel*)sender()); m_pl_manager->activatePlayList((PlayListModel*)sender()); disconnect(sender(), SIGNAL(trackAdded(PlayListTrack*)), this, SLOT(playTrack(PlayListTrack*))); if(!m_pl_manager->currentPlayList()->setCurrent(item)) return; m_core->stop(); m_player->play(); } void Player2Object::disconnectPl() { disconnect(sender(), SIGNAL(trackAdded(PlayListTrack*)), this, SLOT(playTrack(PlayListTrack*))); } void Player2Object::setModel(PlayListModel *selected, PlayListModel *previous) { if(previous) disconnect(previous, 0, this, 0); //disconnect previous model connect(selected, SIGNAL(listChanged()), SLOT(emitPropertiesChanged())); } void Player2Object::syncProperties() { m_props["CanGoNext"] = canGoNext(); m_props["CanGoPrevious"] = canGoPrevious(); m_props["CanPause"] = canPause(); m_props["CanPlay"] = canPlay(); m_props["CanSeek"] = canSeek(); m_props["LoopStatus"] = loopStatus(); m_props["MaximumRate"] = maximumRate(); m_props["MinimumRate"] = minimumRate(); m_props["PlaybackStatus"] = playbackStatus(); m_props["Rate"] = rate(); m_props["Shuffle"] = shuffle(); m_props["Volume"] = volume(); m_props["Metadata"] = metadata(); }