Network remote: Each song will be send after client confirms that the previous track was received. Saves a lot of memory especially when sendings lossless files.

This commit is contained in:
Andreas 2013-07-18 11:28:53 +02:00
parent 4406992554
commit f67083c92e
6 changed files with 56 additions and 3 deletions

View File

@ -16,6 +16,7 @@ enum MsgType {
CLOSE_PLAYLIST = 11; CLOSE_PLAYLIST = 11;
GET_LYRICS = 14; GET_LYRICS = 14;
DOWNLOAD_SONGS = 15; DOWNLOAD_SONGS = 15;
SEND_NEXT_SONG = 16;
// Lastfm // Lastfm
LOVE = 12; LOVE = 12;
BAN = 13; BAN = 13;

View File

@ -149,6 +149,9 @@ void IncomingDataParser::Parse(const pb::remote::Message& msg) {
case pb::remote::DOWNLOAD_SONGS: case pb::remote::DOWNLOAD_SONGS:
emit SendSongs(msg.request_download_songs(), client); emit SendSongs(msg.request_download_songs(), client);
break; break;
case pb::remote::SEND_NEXT_SONG:
emit SendNextSong(client);
break;
default: break; default: break;
} }
} }

View File

@ -45,6 +45,7 @@ signals:
void RemoveSongs(const QList<int>& indices); void RemoveSongs(const QList<int>& indices);
void SeekTo(int seconds); void SeekTo(int seconds);
void SendSongs(const pb::remote::RequestDownloadSongs& request, RemoteClient* client); void SendSongs(const pb::remote::RequestDownloadSongs& request, RemoteClient* client);
void SendNextSong(RemoteClient* client);
private: private:
Application* app_; Application* app_;

View File

@ -163,6 +163,10 @@ void NetworkRemote::AcceptConnection() {
SIGNAL(SendSongs(pb::remote::RequestDownloadSongs,RemoteClient*)), SIGNAL(SendSongs(pb::remote::RequestDownloadSongs,RemoteClient*)),
outgoing_data_creator_.get(), outgoing_data_creator_.get(),
SLOT(SendSongs(pb::remote::RequestDownloadSongs,RemoteClient*))); SLOT(SendSongs(pb::remote::RequestDownloadSongs,RemoteClient*)));
connect(incoming_data_parser_.get(),
SIGNAL(SendNextSong(RemoteClient*)),
outgoing_data_creator_.get(),
SLOT(SendNextSong(RemoteClient*)));
} }
QTcpServer* server = qobject_cast<QTcpServer*>(sender()); QTcpServer* server = qobject_cast<QTcpServer*>(sender());

View File

@ -144,6 +144,8 @@ void OutgoingDataCreator::SendDataToClients(pb::remote::Message* msg) {
if (client->isDownloader()) { if (client->isDownloader()) {
if (client->State() != QTcpSocket::ConnectedState) { if (client->State() != QTcpSocket::ConnectedState) {
clients_->removeAt(clients_->indexOf(client)); clients_->removeAt(clients_->indexOf(client));
delete download_queue_.value(client);
download_queue_.remove(client);
delete client; delete client;
} }
continue; continue;
@ -547,9 +549,20 @@ void OutgoingDataCreator::SendLyrics(int id, const SongInfoFetcher::Result& resu
void OutgoingDataCreator::SendSongs(const pb::remote::RequestDownloadSongs &request, void OutgoingDataCreator::SendSongs(const pb::remote::RequestDownloadSongs &request,
RemoteClient* client) { RemoteClient* client) {
if (!download_queue_.contains(client)) {
download_queue_.insert(client, new QQueue<DownloadItem>());
}
DownloadItem item;
switch (request.download_item()) { switch (request.download_item()) {
case pb::remote::CurrentItem: case pb::remote::CurrentItem:
SendSingleSong(client, current_song_, 1, 1); item.song = current_song_;
item.song_no = 1;
item.song_count = 1;
download_queue_.value(client)->append(item);
break; break;
case pb::remote::ItemAlbum: case pb::remote::ItemAlbum:
SendAlbum(client, current_song_); SendAlbum(client, current_song_);
@ -560,6 +573,19 @@ void OutgoingDataCreator::SendSongs(const pb::remote::RequestDownloadSongs &requ
default: default:
break; break;
} }
// Send first file
SendNextSong(client);
}
void OutgoingDataCreator::SendNextSong(RemoteClient *client) {
if (!download_queue_.contains(client))
return;
// Get the item and send the single song
DownloadItem item;
item = download_queue_.value(client)->dequeue();
SendSingleSong(client, item.song, item.song_no, item.song_count);
} }
void OutgoingDataCreator::SendSingleSong(RemoteClient* client, const Song &song, void OutgoingDataCreator::SendSingleSong(RemoteClient* client, const Song &song,
@ -620,7 +646,11 @@ void OutgoingDataCreator::SendAlbum(RemoteClient *client, const Song &song) {
SongList album = app_->library_backend()->GetSongsByAlbum(song.album()); SongList album = app_->library_backend()->GetSongsByAlbum(song.album());
foreach (Song s, album) { foreach (Song s, album) {
SendSingleSong(client, s, album.indexOf(s)+1, album.size()); DownloadItem item;
item.song = s;
item.song_no = album.indexOf(s)+1;
item.song_count = album.size();
download_queue_.value(client)->append(item);
} }
} }
@ -633,6 +663,10 @@ void OutgoingDataCreator::SendPlaylist(RemoteClient *client, int playlist_id) {
SongList song_list = playlist->GetAllSongs(); SongList song_list = playlist->GetAllSongs();
foreach (Song s, song_list) { foreach (Song s, song_list) {
SendSingleSong(client, s, song_list.indexOf(s)+1, song_list.size()); DownloadItem item;
item.song = s;
item.song_no = song_list.indexOf(s)+1;
item.song_count = song_list.size();
download_queue_.value(client)->append(item);
} }
} }

View File

@ -5,6 +5,8 @@
#include <QImage> #include <QImage>
#include <QList> #include <QList>
#include <QTimer> #include <QTimer>
#include <QMap>
#include <QQueue>
#include "core/player.h" #include "core/player.h"
#include "core/application.h" #include "core/application.h"
@ -26,6 +28,12 @@
typedef QList<SongInfoProvider*> ProviderList; typedef QList<SongInfoProvider*> ProviderList;
struct DownloadItem {
Song song;
int song_no;
int song_count;
};
class OutgoingDataCreator : public QObject { class OutgoingDataCreator : public QObject {
Q_OBJECT Q_OBJECT
public: public:
@ -59,6 +67,7 @@ public slots:
void GetLyrics(); void GetLyrics();
void SendLyrics(int id, const SongInfoFetcher::Result& result); void SendLyrics(int id, const SongInfoFetcher::Result& result);
void SendSongs(const pb::remote::RequestDownloadSongs& request, RemoteClient* client); void SendSongs(const pb::remote::RequestDownloadSongs& request, RemoteClient* client);
void SendNextSong(RemoteClient* client);
private: private:
Application* app_; Application* app_;
@ -70,6 +79,7 @@ private:
QTimer* keep_alive_timer_; QTimer* keep_alive_timer_;
QTimer* track_position_timer_; QTimer* track_position_timer_;
int keep_alive_timeout_; int keep_alive_timeout_;
QMap<RemoteClient*, QQueue<DownloadItem>* > download_queue_;
boost::scoped_ptr<UltimateLyricsReader> ultimate_reader_; boost::scoped_ptr<UltimateLyricsReader> ultimate_reader_;
ProviderList provider_list_; ProviderList provider_list_;