diff --git a/data/ui/library-panel.ui b/data/ui/library-panel.ui
index 3471ce3..7663ac3 100644
--- a/data/ui/library-panel.ui
+++ b/data/ui/library-panel.ui
@@ -15,419 +15,211 @@
-
-
- gtk-refresh
- True
- True
- True
-
-
-
-
-
-
-
- True
- False
- vertical
-
-
- True
- False
- vertical
-
-
-
-
-
- True
- False
- Sort
-
-
-
-
-
- sort by artist
- True
- True
- False
-
- sort_year
-
-
-
-
-
-
- sort by title
- True
- True
- False
-
- sort_year
-
-
-
-
-
-
- sort by year
- True
- True
- False
- True
-
-
-
-
-
-
-
- gtk-sort-descending
- True
- True
- False
-
- True
-
-
-
-
-
-
-
-
-
-
-
+
horizontal
end
6
-
-
- True
- True
- True
- Search the library
-
-
- True
- False
- system-search-symbolic
-
+
+ True
+ Search the library
+
+
+ system-search-symbolic
+
+
+
+
-
-
-
-
-
-
- True
- True
- True
- Select multiple albums
-
-
- True
- False
- object-select-symbolic
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- slide-left-right
-
-
- True
- False
- vertical
-
-
-
-
-
-
+
True
True
-
- search library
-
-
-
-
-
+ True
+ Select multiple albums
+
+
+
+ object-select-symbolic
+
+
+
-
-
-
-
- True
- False
- crossfade
-
-
+
-
-
-
-
- true
-
-
- vertical
-
-
-
-
-
+
+ crossfade
+
+
+ vertical
+
+
+ vertical
+
+
+ center
+ vertical
+ 10
+
+
+ large
+ image-x-generic-symbolic
+
+
+
+
+ 200
+ True
+ False
+ 0
+ True
+
+
+
+
+
+
+
+
+
+
+ true
+
+
+ vertical
+ true
+
+
+
+
+
+ true
vertical
-
contain
@@ -449,202 +241,98 @@
]]>
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+ slide-up
+
+
+
+
+ cancel
+ True
+
+
+
+
+
+ queue
+ True
+
+
+
+
+
+
+
+
+
+
+
+ vertical
+
+
+ true
+ crossfade
+
+
+ True
+ False
+
+
+
+
+ True
+ True
+ False
+ False
+
+
+ True
+ False
+
+
+ gtk-missing-image
+ large
+
+
+
+
+
+
+
+
+
+
+
+
+ play
+ True
+
+
+
+
+
+ queue
+ True
+
+
+
+
+
-
-
-
-
-
-
-
-
-
- none
- false
-
-
-
-
- cancel
- True
- True
- True
-
-
-
-
-
-
- queue
- True
- True
- True
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- True
- False
- vertical
-
-
- True
- False
- crossfade
-
-
- True
- False
-
-
-
-
-
- True
- True
- False
- False
-
-
-
- True
- False
-
-
- True
- False
- gtk-missing-image
-
-
-
-
-
-
-
-
-
-
-
-
-
- True
- False
-
-
- queue
- True
- True
- True
-
-
-
-
-
-
- play
- True
- True
- True
-
-
-
-
-
-
-
-
-
-
diff --git a/data/ui/playlist-panel.ui b/data/ui/playlist-panel.ui
index fc4c094..8796afc 100644
--- a/data/ui/playlist-panel.ui
+++ b/data/ui/playlist-panel.ui
@@ -53,7 +53,7 @@
true
@@ -93,7 +93,7 @@
@@ -160,16 +160,16 @@
- remove
+ play
True
-
+
- play
+ remove
True
-
+
diff --git a/src/client.py b/src/client.py
index b2a5d69..0009b54 100644
--- a/src/client.py
+++ b/src/client.py
@@ -616,17 +616,16 @@ class Client(Base):
def _queue_album(self, album):
track_ids = []
- if album in self._albums:
- self._logger.info("add album %s", album)
- for track in self._albums[album].get_tracks():
- self._logger.info("addid: %r", track.get_file())
- track_id = None
- track_id_response = self._parse_dict(self._call('addid', track.get_file()))
- if 'id' in track_id_response:
- track_id = track_id_response['id']
- self._logger.debug("track id: %r", track_id)
- if track_id is not None:
- track_ids.append(track_id)
+ self._logger.info("add album %s", album)
+ for track in album.get_tracks():
+ self._logger.info("addid: %r", track.get_file())
+ track_id = None
+ track_id_response = self._parse_dict(self._call('addid', track.get_file()))
+ if 'id' in track_id_response:
+ track_id = track_id_response['id']
+ self._logger.debug("track id: %r", track_id)
+ if track_id is not None:
+ track_ids.append(track_id)
return track_ids
@@ -1093,7 +1092,7 @@ class MCGAlbum:
return True
- def compare(album1, album2, criterion=None):
+ def compare(album1, album2, criterion=None, reverse=False):
if criterion == None:
criterion = SortOrder.TITLE
if criterion == SortOrder.ARTIST:
@@ -1103,20 +1102,22 @@ class MCGAlbum:
elif criterion == SortOrder.YEAR:
value_function = "get_date"
+ reverseMultiplier = -1 if reverse else 1
+
value1 = getattr(album1, value_function)()
value2 = getattr(album2, value_function)()
if value1 is None and value2 is None:
return 0
elif value1 is None:
- return -1
+ return -1 * reverseMultiplier
elif value2 is None:
- return 1
+ return 1 * reverseMultiplier
if value1 < value2:
- return -1
+ return -1 * reverseMultiplier
elif value1 == value2:
return 0
else:
- return 1
+ return 1 * reverseMultiplier
diff --git a/src/librarypanel.py b/src/librarypanel.py
index 79c85f6..11b68f5 100644
--- a/src/librarypanel.py
+++ b/src/librarypanel.py
@@ -16,6 +16,7 @@ from mcg.albumheaderbar import AlbumHeaderbar
from mcg.utils import SortOrder
from mcg.utils import Utils
from mcg.utils import GridItem
+from mcg.utils import SearchFilter
@@ -32,20 +33,18 @@ class LibraryPanel(Adw.Bin):
'queue-multiple': (GObject.SIGNAL_RUN_FIRST, None, (GObject.TYPE_PYOBJECT,)),
'item-size-changed': (GObject.SIGNAL_RUN_FIRST, None, (int,)),
'sort-order-changed': (GObject.SIGNAL_RUN_FIRST, None, (int,)),
- 'sort-type-changed': (GObject.SIGNAL_RUN_FIRST, None, (Gtk.SortType,)),
+ 'sort-type-changed': (GObject.SIGNAL_RUN_FIRST, None, (bool,)),
'albumart': (GObject.SIGNAL_RUN_FIRST, None, (str,)),
- 'select': (GObject.SIGNAL_RUN_FIRST, None, (bool,)),
- 'toggle-search': (GObject.SIGNAL_RUN_FIRST, None, (bool,)),
'update': (GObject.SIGNAL_RUN_FIRST, None, ()),
'start-scale': (GObject.SIGNAL_RUN_FIRST, None, (int,)),
'end-scale': (GObject.SIGNAL_RUN_FIRST, None, (int,)),
- 'sort': (GObject.SIGNAL_RUN_FIRST, None, (int,)),
- 'sort-type': (GObject.SIGNAL_RUN_FIRST, None, (Gtk.SortType,))
}
# Widgets
+ library_stack = Gtk.Template.Child()
+ panel_normal = Gtk.Template.Child()
panel_standalone = Gtk.Template.Child()
actionbar_revealer = Gtk.Template.Child()
# Toolbar
@@ -69,6 +68,9 @@ class LibraryPanel(Adw.Bin):
scroll = Gtk.Template.Child()
# Library Grid
library_grid = Gtk.Template.Child()
+ # Action bar (normal)
+ actionbar = Gtk.Template.Child()
+ actionbar_standalone = Gtk.Template.Child()
# Standalone Image
standalone_stack = Gtk.Template.Child()
standalone_spinner = Gtk.Template.Child()
@@ -83,7 +85,6 @@ class LibraryPanel(Adw.Bin):
self._buttons = {}
self._albums = None
self._host = "localhost"
- self._filter_string = ""
self._item_size = 150
self._sort_order = SortOrder.YEAR
self._sort_type = Gtk.SortType.DESCENDING
@@ -99,22 +100,19 @@ class LibraryPanel(Adw.Bin):
# Widgets
# FIXME Toolbar signals
- #self._toolbar.connect('select', self.on_toolbar_select)
- #self._toolbar.connect('toggle-search', self.on_toolbar_toggle_search)
- #self._toolbar.connect('update', self.on_toolbar_update)
#self._toolbar.connect('start-scale', self.on_toolbar_scale)
#self._toolbar.connect('end-scale', self.on_toolbar_scaled)
- #self._toolbar.connect('sort', self.on_toolbar_sort)
- #self._toolbar.connect('sort-type', self.on_toolbar_sort_type)
# Header bar
self._headerbar_standalone = AlbumHeaderbar()
self._headerbar_standalone.connect('close', self.on_standalone_close_clicked)
# Library Grid: Model
self._library_grid_model = Gio.ListStore()
- self._library_grid_selection = Gtk.MultiSelection.new(self._library_grid_model)
- self._library_grid_filter = self._library_grid_selection
+ self._library_grid_filter = Gtk.FilterListModel()
+ self._library_grid_filter.set_model(self._library_grid_model)
+ self._library_grid_selection_multi = Gtk.MultiSelection.new(self._library_grid_filter)
+ self._library_grid_selection_single = Gtk.SingleSelection.new(self._library_grid_filter)
# Library Grid
- self.library_grid.set_model(self._library_grid_filter)
+ self.library_grid.set_model(self._library_grid_selection_single)
# Toolbar menu
self.grid_scale.set_value(self._item_size)
self._toolbar_sort_buttons = {
@@ -136,17 +134,21 @@ class LibraryPanel(Adw.Bin):
self._is_selected = selected
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_select_toggled(self, widget):
- self.emit('select', widget.get_active())
+ if self.select_button.get_active():
+ self.actionbar_revealer.set_reveal_child(True)
+ self.library_grid.set_model(self._library_grid_selection_multi)
+ self.library_grid.set_single_click_activate(False)
+ self.library_grid.get_style_context().add_class(Utils.CSS_SELECTION)
+ else:
+ self.actionbar_revealer.set_reveal_child(False)
+ self.library_grid.set_model(self._library_grid_selection_single)
+ self.library_grid.set_single_click_activate(True)
+ self.library_grid.get_style_context().remove_class(Utils.CSS_SELECTION)
- #@Gtk.Template.Callback()
- def on_search_toggled(self, widget):
- self.emit('toggle-search', widget.get_active())
-
-
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_update_clicked(self, widget):
self.emit('update')
@@ -164,20 +166,22 @@ class LibraryPanel(Adw.Bin):
self.toolbar_popover.popdown()
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_sort_toggled(self, widget):
if widget.get_active():
- sort = [key for key, value in self._toolbar_sort_buttons.items() if value is widget][0]
- self.emit('sort', sort)
+ self._sort_order = [key for key, value in self._toolbar_sort_buttons.items() if value is widget][0]
+ self._sort_grid_model()
+ self.emit('sort-order-changed', self._sort_order)
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_sort_order_toggled(self, button):
if button.get_active():
- sort_type = Gtk.SortType.DESCENDING
+ self._sort_type = Gtk.SortType.DESCENDING
else:
- sort_type = Gtk.SortType.ASCENDING
- self.emit('sort-type', sort_type)
+ self._sort_type = Gtk.SortType.ASCENDING
+ self._sort_grid_model()
+ self.emit('sort-type-changed', button.get_active())
# FIXME on_resize()
@@ -204,21 +208,6 @@ class LibraryPanel(Adw.Bin):
)
- def on_toolbar_toggle_search(self, widget, active):
- self.filter_bar.set_search_mode(active)
-
-
- def on_toolbar_select(self, widget, active):
- if active:
- self.actionbar_revealer.set_reveal_child(True)
- self.library_grid.set_selection_mode(Gtk.SelectionMode.MULTIPLE)
- self.library_grid.get_style_context().add_class(Utils.CSS_SELECTION)
- else:
- self.actionbar_revealer.set_reveal_child(False)
- self.library_grid.set_selection_mode(Gtk.SelectionMode.SINGLE)
- self.library_grid.get_style_context().remove_class(Utils.CSS_SELECTION)
-
-
def on_toolbar_update(self, widget):
self.emit('update')
@@ -243,41 +232,22 @@ class LibraryPanel(Adw.Bin):
return False
- def on_toolbar_sort(self, widget, sort):
- self._change_sort(sort)
-
-
- def on_toolbar_sort_type(self, widget, sort_type):
- self._sort_type = sort_type
- #self._library_grid_model.set_sort_column_id(2, sort_type)
- self.emit('sort-type-changed', sort_type)
-
-
- #@Gtk.Template.Callback()
- def on_filter_bar_notify(self, widget, value):
- if self.toolbar_search_bar.get_active() is not self.filter_bar.get_search_mode():
- self.toolbar_search_bar.set_active(self.filter_bar.get_search_mode())
-
-
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_filter_entry_changed(self, widget):
- self._filter_string = self.filter_entry.get_text()
- GObject.idle_add(self._library_grid_filter.refilter)
+ self._library_grid_filter.set_filter(SearchFilter(self.filter_entry.get_text()))
- # FIXME on_library_grid_clicked()
- #@Gtk.Template.Callback()
- def on_library_grid_clicked(self, widget, path):
+ @Gtk.Template.Callback()
+ def on_library_grid_clicked(self, widget, position):
# Get selected album
- path = self._library_grid_filter.convert_path_to_child_path(path)
- iter = self._library_grid_model.get_iter(path)
- id = self._library_grid_model.get_value(iter, 2)
- album = self._albums[id]
+ item = self._library_grid_filter.get_item(position)
+ album = item.get_album()
+ id = album.get_id()
self._selected_albums = [album]
self.emit('albumart', id)
# Show standalone album
- if widget.get_selection_mode() == Gtk.SelectionMode.SINGLE:
+ if widget.get_model() == self._library_grid_selection_single:
# Set labels
self._headerbar_standalone.set_album(album)
@@ -289,34 +259,14 @@ class LibraryPanel(Adw.Bin):
self.standalone_spinner.start()
- # FIXME on_library_grid_selection_changed()
- #@Gtk.Template.Callback()
- def on_library_grid_selection_changed(self, widget):
- self._selected_albums = []
- for path in widget.get_selected_items():
- path = self._library_grid_filter.convert_path_to_child_path(path)
- iter = self._library_grid_model.get_iter(path)
- id = self._library_grid_model.get_value(iter, 2)
- self._selected_albums.insert(0, self._albums[id])
-
-
- def on_filter_visible(self, model, iter, data):
- id = model.get_value(iter, 2)
- if not id in self._albums.keys():
- return
- album = self._albums[id]
- return album.filter(self._filter_string)
-
-
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_selection_cancel_clicked(self, widget):
self.select_button.set_active(False)
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_selection_add_clicked(self, widget):
- ids = [album.get_id() for album in self._selected_albums]
- self.emit('queue-multiple', ids)
+ self.emit('queue-multiple', self._get_selected_albums())
self.select_button.set_active(False)
@@ -326,13 +276,13 @@ class LibraryPanel(Adw.Bin):
self._resize_standalone_image()
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_standalone_play_clicked(self, widget):
self.emit('play', self._selected_albums[0].get_id())
self._close_standalone()
- #@Gtk.Template.Callback()
+ @Gtk.Template.Callback()
def on_standalone_queue_clicked(self, widget):
self.emit('queue', self._selected_albums[0].get_id())
self._close_standalone()
@@ -358,29 +308,21 @@ class LibraryPanel(Adw.Bin):
def set_sort_order(self, sort):
- if self._sort_order != sort:
- button = self._toolbar_sort_buttons[sort]
- if button and not button.get_active():
+ button = self._toolbar_sort_buttons[sort]
+ if button:
+ self._sort_order = [key for key, value in self._toolbar_sort_buttons.items() if value is button][0]
+ if not button.get_active():
button.set_active(True)
- self._sort_order = sort
- self._library_grid_model.set_sort_func(2, self.compare_albums, self._sort_order)
-
-
- def get_sort_order(self):
- return self._sort_order
+ self._sort_grid_model()
def set_sort_type(self, sort_type):
- if self._sort_type != sort_type:
- if sort_type:
- sort_type_gtk = Gtk.SortType.DESCENDING
- self.toolbar_sort_order_button.set_active(True)
- else:
- sort_type_gtk = Gtk.SortType.ASCENDING
- self.toolbar_sort_order_button.set_active(False)
- if self._sort_type != sort_type_gtk:
- self._sort_type = sort_type_gtk
- self._library_grid_model.set_sort_column_id(2, sort_type)
+ sort_type_gtk = Gtk.SortType.DESCENDING if sort_type else Gtk.SortType.ASCENDING
+
+ if sort_type_gtk != self._sort_type:
+ self._sort_type = sort_type_gtk
+ self.toolbar_sort_order_button.set_active(sort_type)
+ self._sort_grid_model()
def get_sort_type(self):
@@ -416,25 +358,18 @@ class LibraryPanel(Adw.Bin):
GObject.idle_add(self._show_image)
- def compare_albums(self, model, row1, row2, criterion):
- id1 = model.get_value(row1, 2)
- id2 = model.get_value(row2, 2)
+ def _sort_grid_model(self):
+ self._library_grid_model.sort(self._grid_model_compare_func, self._sort_order, self._sort_type)
- if not id1 or not id2:
- return
- return client.MCGAlbum.compare(self._albums[id1], self._albums[id2], criterion)
+
+ def _grid_model_compare_func(self, item1, item2, criterion, order):
+ return client.MCGAlbum.compare(item1.get_album(), item2.get_album(), criterion, (order == Gtk.SortType.DESCENDING))
def stop_threads(self):
self._library_stop.set()
- def _change_sort(self, sort):
- self._sort_order = sort
- self._library_grid_model.set_sort_func(2, self.compare_albums, sort)
- self.emit('sort-order-changed', sort)
-
-
def _set_albums(self, host, albums, size):
"""
if not self._is_selected and albums != self._albums:
@@ -451,6 +386,7 @@ class LibraryPanel(Adw.Bin):
GObject.idle_add(self.stack.set_visible_child, self.progress_box)
GObject.idle_add(self.progress_bar.set_fraction, 0.0)
#GObject.idle_add(self.library_grid.set_item_padding, size / 100)
+ temp_model = self.library_grid.get_model()
self.library_grid.set_model(None)
#self.library_grid.freeze_child_notify()
self._library_grid_model.remove_all()
@@ -489,11 +425,12 @@ class LibraryPanel(Adw.Bin):
self._library_lock.release()
return
- self.library_grid.set_model(self._library_grid_filter)
+ self.library_grid.set_model(temp_model)
#self.library_grid.thaw_child_notify()
#self.library_grid.set_item_width(-1)
self._library_lock.release()
self.stack.set_visible_child(self.scroll)
+ self._sort_grid_model()
def _set_widget_grid_size(self, grid_widget, size, vertical):
@@ -563,12 +500,12 @@ class LibraryPanel(Adw.Bin):
def _open_standalone(self):
- self.set_visible_child(self.panel_standalone)
+ self.library_stack.set_visible_child(self.panel_standalone)
self.emit('open-standalone')
def _close_standalone(self):
- self.set_visible_child(self.get_children()[0])
+ self.library_stack.set_visible_child(self.panel_normal)
self.emit('close-standalone')
@@ -594,7 +531,6 @@ class LibraryPanel(Adw.Bin):
if width <= 0 or height <= 0:
return
# Pixelpuffer auf Oberfläche zeichnen
- self.standalone_image.set_allocation(self.standalone_scroll.get_allocation())
self.standalone_image.set_from_pixbuf(pixbuf.scale_simple(width, height, GdkPixbuf.InterpType.HYPER))
self.standalone_image.show()
@@ -608,3 +544,11 @@ class LibraryPanel(Adw.Bin):
Gtk.TextDirection.LTR,
Gtk.IconLookupFlags.FORCE_SYMBOLIC
)
+
+
+ def _get_selected_albums(self):
+ albums = []
+ for i in range(self.library_grid.get_model().get_n_items()):
+ if self.library_grid.get_model().is_selected(i):
+ albums.append(self.library_grid.get_model().get_item(i).get_album())
+ return albums
diff --git a/src/utils.py b/src/utils.py
index f5ad22e..e67655d 100644
--- a/src/utils.py
+++ b/src/utils.py
@@ -8,7 +8,7 @@ import locale
import os
import urllib
-from gi.repository import Gdk, GdkPixbuf, GObject
+from gi.repository import Gdk, GdkPixbuf, GObject, Gtk
@@ -112,3 +112,17 @@ class GridItem(GObject.GObject):
def get_album(self):
return self._album
+
+
+
+
+class SearchFilter(Gtk.Filter):
+
+
+ def __init__(self, search_string):
+ super().__init__()
+ self._search_string = search_string
+
+
+ def do_match(self, grid_item):
+ return grid_item.get_album().filter(self._search_string)
diff --git a/src/window.py b/src/window.py
index 84aae61..d915047 100644
--- a/src/window.py
+++ b/src/window.py
@@ -105,8 +105,8 @@ class Window(Adw.ApplicationWindow):
self._panels.append(self._playlist_panel)
# Library panel
self._library_panel = LibraryPanel(self._mcg)
- #self._library_panel.connect('open-standalone', self.on_panel_open_standalone)
- #self._library_panel.connect('close-standalone', self.on_panel_close_standalone)
+ self._library_panel.connect('open-standalone', self.on_panel_open_standalone)
+ self._library_panel.connect('close-standalone', self.on_panel_close_standalone)
self._panels.append(self._library_panel)
# Stack
self.content_stack.add_child(self._connection_panel)
@@ -309,6 +309,7 @@ class Window(Adw.ApplicationWindow):
self.toolbar_view.add_top_bar(self.headerbar)
self.toolbar_view.remove(panel.get_headerbar_standalone())
+
def on_connection_panel_connection_changed(self, widget, host, port, password):
self._settings.set_string(Window.SETTING_HOST, host)
self._settings.set_int(Window.SETTING_PORT, port)
@@ -381,11 +382,11 @@ class Window(Adw.ApplicationWindow):
def on_library_panel_sort_order_changed(self, widget, sort_order):
- self._settings.set_enum(Window.SETTING_SORT_ORDER, self._library_panel.get_sort_order())
+ self._settings.set_enum(Window.SETTING_SORT_ORDER, sort_order)
def on_library_panel_sort_type_changed(self, widget, sort_type):
- self._settings.set_boolean(Window.SETTING_SORT_TYPE, self._library_panel.get_sort_type())
+ self._settings.set_boolean(Window.SETTING_SORT_TYPE, sort_type)
def on_library_panel_albumart(self, widget, album):