[gnome-music/wip/jfelder/mpris-limit-get-songs: 4/4] player: Handle repeat and limit songs in get_songs
- From: Jean Felder <jfelder src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-music/wip/jfelder/mpris-limit-get-songs: 4/4] player: Handle repeat and limit songs in get_songs
- Date: Sun, 28 Oct 2018 22:10:00 +0000 (UTC)
commit 375efba9abefc339c660ebbfb8ff5ed0bb1804c3
Author: Jean Felder <jfelder src gnome org>
Date: Fri Sep 14 00:16:17 2018 +0200
player: Handle repeat and limit songs in get_songs
get_songs method is used by MPRIS to expose a playlist to MPRIS
clients. Returning the whole playlist (for example, all the songs) can
create instabilities and crashes.
Besides, according to MPRIS specification, TrackList should only be
a "short list of tracks which were recently played or will be played
shortly."
Limit the number of returned to avoid those instabilities. Order the
returned list following the repeat mode to reflect the current state
of the player playlist.
Update MPRIS code to reflect these changes.
gnomemusic/mpris.py | 28 +++++++++-----
gnomemusic/player.py | 107 +++++++++++++++++++++++++++++++++++++++------------
2 files changed, 100 insertions(+), 35 deletions(-)
---
diff --git a/gnomemusic/mpris.py b/gnomemusic/mpris.py
index 41f69005..1d037ec5 100644
--- a/gnomemusic/mpris.py
+++ b/gnomemusic/mpris.py
@@ -358,6 +358,7 @@ class MediaPlayer2Service(Server):
@log
def _update_songs_list(self):
+ previous_path_list = self._path_list
self._path_list = []
self._metadata_list = []
for song in self.player.get_songs():
@@ -366,6 +367,16 @@ class MediaPlayer2Service(Server):
self._path_list.append(path)
self._metadata_list.append(metadata)
+ if (not previous_path_list
+ or previous_path_list[0] != self._path_list[0]
+ or previous_path_list[-1] != self._path_list[-1]):
+ current_song_path = self._get_song_dbus_path(
+ self.player.props.current_song)
+ self.TrackListReplaced(self._path_list, current_song_path)
+ self.PropertiesChanged(
+ MediaPlayer2Service.MEDIA_PLAYER2_TRACKLIST_IFACE,
+ {'Tracks': GLib.Variant('ao', self._path_list), }, [])
+
@log
def _get_playlist_dbus_path(self, playlist):
"""Convert a playlist to a D-Bus path
@@ -429,6 +440,8 @@ class MediaPlayer2Service(Server):
@log
def _on_current_song_changed(self, player, position):
+ self._update_songs_list()
+
if self.player.props.repeat_mode == RepeatMode.SONG:
self.Seeked(0)
@@ -490,14 +503,6 @@ class MediaPlayer2Service(Server):
def _on_player_playlist_changed(self, klass):
self._update_songs_list()
- if self.player.props.current_song:
- current_song_path = self._get_song_dbus_path(
- self.player.props.current_song)
- self.TrackListReplaced(self._path_list, current_song_path)
- self.PropertiesChanged(
- MediaPlayer2Service.MEDIA_PLAYER2_TRACKLIST_IFACE,
- {'Tracks': GLib.Variant('ao', self._path_list), }, [])
-
if (self.player.get_playlist_type() == PlayerPlaylist.Type.PLAYLIST
or self._player_previous_type == PlayerPlaylist.Type.PLAYLIST):
variant = GLib.Variant('(b(oss))', self._get_active_playlist())
@@ -609,8 +614,11 @@ class MediaPlayer2Service(Server):
pass
def GoTo(self, path):
- index = self.path_list.index(path)
- self.player.play(index)
+ current_song_path = self._get_song_dbus_path(
+ self.player.props.current_song)
+ current_song_index = self._path_list.index(current_song_path)
+ goto_index = self._path_list.index(path)
+ self.player.play(goto_index - current_song_index)
return
def TrackListReplaced(self, tracks, current_song):
diff --git a/gnomemusic/player.py b/gnomemusic/player.py
index 32bce1b5..9a0bfe83 100644
--- a/gnomemusic/player.py
+++ b/gnomemusic/player.py
@@ -86,6 +86,8 @@ class PlayerPlaylist(GObject.GObject):
'song-validated': (GObject.SignalFlags.RUN_FIRST, None, (int, int)),
}
+ _nb_songs_max = 10
+
def __repr__(self):
return '<PlayerPlayList>'
@@ -160,13 +162,14 @@ class PlayerPlaylist(GObject.GObject):
return True
@log
- def set_song(self, song_index):
+ def set_song(self, song_offset):
"""Change playlist index.
:param int song_index: requested song index
:return: True if the index has changed. False otherwise.
:rtype: bool
"""
+ song_index = song_offset + self._current_index
if song_index >= len(self._songs):
return False
@@ -287,36 +290,42 @@ class PlayerPlaylist(GObject.GObject):
self._discoverer.discover_uri_async(url)
@log
- def _get_next_index(self):
- if not self.has_next():
+ def _get_next_index(self, index=None):
+ if index is None:
+ index = self._current_index
+
+ if not self.has_next(index):
return -1
if self._repeat == RepeatMode.SONG:
- return self._current_index
+ return index
if (self._repeat == RepeatMode.ALL
- and self._current_index == (len(self._songs) - 1)):
+ and index == (len(self._songs) - 1)):
return 0
if self._repeat == RepeatMode.SHUFFLE:
- index = self._shuffle_indexes.index(self._current_index)
- return self._shuffle_indexes[index + 1]
+ shuffle_index = self._shuffle_indexes.index(index)
+ return self._shuffle_indexes[shuffle_index + 1]
else:
- return self._current_index + 1
+ return index + 1
@log
- def _get_previous_index(self):
- if not self.has_previous():
+ def _get_previous_index(self, index=None):
+ if index is None:
+ index = self._current_index
+
+ if not self.has_previous(index):
return -1
if self._repeat == RepeatMode.SONG:
- return self._current_index
+ return index
if (self._repeat == RepeatMode.ALL
- and self._current_index == 0):
+ and index == 0):
return len(self._songs) - 1
if self._repeat == RepeatMode.SHUFFLE:
- index = self._shuffle_indexes.index(self._current_index)
- return self._shuffle_indexes[index - 1]
+ shuffle_index = self._shuffle_indexes.index(index)
+ return self._shuffle_indexes[shuffle_index - 1]
else:
- return self._current_index - 1
+ return index - 1
@log
def _validate_next_song(self):
@@ -337,27 +346,32 @@ class PlayerPlaylist(GObject.GObject):
self._validate_song(previous_index)
@log
- def has_next(self):
+ def has_next(self, index=None):
"""Test if there is a song after the current one.
:return: True if there is a song. False otherwise.
:rtype: bool
"""
+ if index is None:
+ index = self._current_index
if (self._repeat == RepeatMode.SHUFFLE
and self._shuffle_indexes):
- index = self._shuffle_indexes.index(self._current_index)
+ index = self._shuffle_indexes.index(index)
return index < (len(self._shuffle_indexes) - 1)
if self._repeat != RepeatMode.NONE:
return True
- return self._current_index < (len(self._songs) - 1)
+ return index < (len(self._songs) - 1)
@log
- def has_previous(self):
+ def has_previous(self, index=None):
"""Test if there is a song before the current one.
:return: True if there is a song. False otherwise.
:rtype: bool
"""
+ if index is None:
+ index = self._current_index
+
if (self._repeat == RepeatMode.SHUFFLE
and self._shuffle_indexes):
index = self._shuffle_indexes.index(self._current_index)
@@ -478,12 +492,53 @@ class PlayerPlaylist(GObject.GObject):
@log
def get_songs(self):
- """Get the current playlist.
+ """Get recent and next songs from the current playlist.
+
+ If the playlist is an album, return all songs.
+ Returned songs are sorted according to the repeat mode.
:returns: current playlist
:rtype: list of Grl.Media
"""
- songs = [elt[PlayerField.SONG] for elt in self._songs]
+ if not self.props.current_song:
+ return []
+
+ # if self._repeat == RepeatMode.SONG:
+ # current_song = self._songs[self._current_index][PlayerField.SONG]
+ # return [current_song]
+ songs = []
+
+ if self._type == PlayerPlaylist.Type.ALBUM:
+ first_index = 0
+ last_index = len(self._songs)
+ else:
+ first_index = self._current_index - self._nb_songs_max
+ last_index = self._current_index + self._nb_songs_max
+
+ if self._repeat == RepeatMode.SONG:
+ first_index = max(0, first_index)
+ last_index = min(last_index, len(self._songs))
+ for index in range(first_index, last_index):
+ songs.append(self._songs[index][PlayerField.SONG])
+ return songs
+
+ index_max = min(0, last_index)
+ index = index_max
+ for i in range(first_index, index_max):
+ index = self._get_previous_index(index)
+ if index == -1:
+ break
+ songs.insert(0, self._songs[index][PlayerField.SONG])
+
+ index_min = max(index_max, first_index)
+ songs.append(self._songs[index_min][PlayerField.SONG])
+ index = index_min
+ for i in range(index_min, last_index):
+ index = self._get_next_index(index)
+ if index == -1:
+ break
+ songs.append(self._songs[index][PlayerField.SONG])
+
return songs
@@ -593,13 +648,13 @@ class Player(GObject.GObject):
self.stop()
@log
- def play(self, song_index=None):
+ def play(self, song_offset=None):
"""Play"""
if self.props.current_song is None:
return
- if (song_index is not None
- and not self._playlist.set_song(song_index)):
+ if (song_offset is not None
+ and not self._playlist.set_song(song_offset)):
return False
if self.props.state != Playback.PAUSED:
@@ -835,7 +890,9 @@ class Player(GObject.GObject):
@log
def get_songs(self):
- """Get the current playlist.
+ """Get recent and next songs from the current playlist.
+
+ Returned songs are sorted according to the repeat mode.
:returns: current playlist
:rtype: list of Grl.Media
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]