From 906496716cc310b57e4d63217ed70012b1fd506c Mon Sep 17 00:00:00 2001 From: Christian Heller <c.heller@plomlompom.de> Date: Mon, 23 Sep 2024 05:54:17 +0200 Subject: [PATCH] Add "by first sorter" option to tabelize gallery by attributes. --- browser.py | 238 +++++++++++++++++++++++++++++++++++------------------ 1 file changed, 158 insertions(+), 80 deletions(-) diff --git a/browser.py b/browser.py index f74e0b7..e1463d6 100755 --- a/browser.py +++ b/browser.py @@ -66,8 +66,57 @@ class Sorter(GObject.GObject): self.list_item.get_first_child().set_text(label) -class FileItem(GObject.GObject): +class GallerySlot(Gtk.Button): + """Slot in Gallery representing a GalleryItem.""" + + def __init__(self, item, on_click_file=None): + super().__init__() + self.add_css_class('slot') + self.set_hexpand(True) + self.item = item + self.item.slot = self + if on_click_file: + self.connect('clicked', on_click_file) + + def mark(self, css_class, do_add=True): + """Add or remove css_class from self.""" + if do_add: + self.add_css_class(css_class) + else: + self.remove_css_class(css_class) + + def update_widget(self, slot_size, margin, is_in_vp): + """(Un-)Load content if (not) is_in_vp, update geometry, CSS classes""" + new_content = None + if isinstance(self.item, ImgItem): + if is_in_vp and not isinstance(self.item, Gtk.Image): + new_content = Gtk.Image.new_from_file(self.item.full_path) + if self.item.with_others: + new_content.set_vexpand(True) + box = Gtk.Box(orientation=OR_V) + box.append(new_content) + msg = 'and one or more other images of this configuration' + box.append(Gtk.Label(label=msg)) + new_content = box + elif (not is_in_vp) and not isinstance(self.item, Gtk.Label): + new_content = Gtk.Label(label='?') + elif self.get_child() is None: + label = self.item.name if isinstance(self.item, DirItem) else '+' + new_content = Gtk.Label(label=label) + if new_content: + self.set_child(new_content) + side_margin = margin // 2 + if side_margin: + for s in ('bottom', 'top', 'start', 'end'): + setattr(self.get_child().props, f'margin_{s}', side_margin) + self.get_child().set_size_request(slot_size, slot_size) + if isinstance(self.item, ImgItem): + self.mark('bookmarked', self.item.bookmarked) + + +class GalleryItem(GObject.GObject): """Gallery representation of filesystem entry, base to DirItem, ImgItem.""" + slot: GallerySlot def __init__(self, path, name): super().__init__() @@ -75,7 +124,7 @@ class FileItem(GObject.GObject): self.full_path = path_join(path, self.name) -class DirItem(FileItem): +class DirItem(GalleryItem): """Gallery representation of filesystem entry for directory.""" def __init__(self, path, name, is_parent=False): @@ -84,13 +133,14 @@ class DirItem(FileItem): self.full_path = path -class ImgItem(FileItem): +class ImgItem(GalleryItem): """Gallery representation of filesystem entry for image file.""" def __init__(self, path, name, last_mod_time, cache): super().__init__(path, name) self.last_mod_time = last_mod_time self.bookmarked = False + self.with_others = False for param_name in GEN_PARAMS: if param_name in GEN_PARAMS_STR: setattr(self, param_name.lower(), '') @@ -122,47 +172,8 @@ class ImgItem(FileItem): self.slot.mark('bookmarked', positive) -class GallerySlot(Gtk.Button): - """Slot in Gallery representing a FileItem.""" - - def __init__(self, item, on_click_file): - super().__init__() - self.add_css_class('slot') - self.set_hexpand(True) - self.item = item - self.item.slot = self - self.connect('clicked', on_click_file) - - def mark(self, css_class, do_add=True): - """Add or remove css_class from self.""" - if do_add: - self.add_css_class(css_class) - else: - self.remove_css_class(css_class) - - def update_widget(self, slot_size, margin, is_in_vp): - """(Un-)Load content if (not) is_in_vp, update geometry, CSS classes""" - new_content = None - if self.get_child() is None and isinstance(self.item, DirItem): - new_content = Gtk.Label(label=self.item.name) - elif isinstance(self.item, ImgItem): - if is_in_vp and not isinstance(self.item, Gtk.Image): - new_content = Gtk.Image.new_from_file(self.item.full_path) - elif (not is_in_vp) and not isinstance(self.item, Gtk.Label): - new_content = Gtk.Label(label='?') - if new_content: - self.set_child(new_content) - side_margin = margin // 2 - if side_margin: - for s in ('bottom', 'top', 'start', 'end'): - setattr(self.get_child().props, f'margin_{s}', side_margin) - self.get_child().set_size_request(slot_size, slot_size) - if isinstance(self.item, ImgItem): - self.mark('bookmarked', self.item.bookmarked) - - class Gallery: - """Representation of FileItems below a directory.""" + """Representation of GalleryItems below a directory.""" def __init__(self, sort_order, @@ -177,12 +188,14 @@ class Gallery: self._on_selection_change = on_selection_change self.show_dirs = False + self.per_row_by_first_sorter = False self.per_row = GALLERY_PER_ROW_DEFAULT self._slot_margin = GALLERY_SLOT_MARGIN self._grid = None self._force_width, self._force_height = 0, 0 self.slots = None self.dir_entries = [] + self.dir_entries_filtered_sorted = [] self.selected_idx = 0 self._fixed_frame = Gtk.Fixed(hexpand=True, vexpand=True) @@ -225,8 +238,9 @@ class Gallery: if unselect_old: self.slots[self.selected_idx].mark('selected', False) self.selected_idx = new_idx - self.slots[self.selected_idx].mark('selected', True) - self.slots[self.selected_idx].grab_focus() + if self.slots: + self.slots[self.selected_idx].mark('selected', True) + self.slots[self.selected_idx].grab_focus() self._on_selection_change() def build_and_show(self, suggested_selection=None): @@ -238,22 +252,68 @@ class Gallery: self._on_hit_item() return f + def build_rows_by_attrs(remaining_attrs, items_of_parent_attr_value): + if not items_of_parent_attr_value: + return + attr_name, attr_values = remaining_attrs[0] + if 1 == len(remaining_attrs): + row = [None] * len(attr_values) + for item in items_of_parent_attr_value: + item_val = getattr(item, attr_name) + idx_item_val_in_attr_values = attr_values.index(item_val) + if row[idx_item_val_in_attr_values]: + item.with_others = True + row[idx_item_val_in_attr_values] = item + for i_col, item in enumerate(row): + if item: + slot = GallerySlot(item, item_clicker(i_slot_ref[0])) + else: + slot = GallerySlot(GalleryItem('', '')) # dummy + self.slots += [slot] + i_slot_ref[0] += 1 + self._grid.attach(slot, i_col, i_row_ref[0], 1, 1) + i_row_ref[0] += 1 + return + for attr_value in attr_values: + items_of_attr_value = [x for x in items_of_parent_attr_value + if attr_value == getattr(x, attr_name)] + build_rows_by_attrs(remaining_attrs[1:], items_of_attr_value) + if self._grid: self._fixed_frame.remove(self._grid) self.slots = [] self._grid = Gtk.Grid() self._fixed_frame.put(self._grid, 0, 0) - i_row, i_col = 0, 0 - for i, item in enumerate(sorted([entry for entry in self.dir_entries - if self._filter_func(entry)], - key=cmp_to_key(self._sort_cmp))): - if self.per_row == i_col: - i_col = 0 - i_row += 1 - slot = GallerySlot(item, item_clicker(i)) - self._grid.attach(slot, i_col, i_row, 1, 1) - self.slots += [slot] - i_col += 1 + entries_filtered = [entry for entry in self.dir_entries + if self._filter_func(entry)] + if self.per_row_by_first_sorter: + self.show_dirs = False + sort_attrs = [] + for sorter in reversed(self._sort_order): + values = set() + for item in [x for x in entries_filtered + if isinstance(x, ImgItem)]: + val = None + if hasattr(item, sorter.name): + val = getattr(item, sorter.name) + values.add(val) + sort_attrs += [(sorter.name, sorted(list(values)))] + i_row_ref = [0] + i_slot_ref = [0] + self.per_row = len(sort_attrs[-1][1]) + build_rows_by_attrs(sort_attrs, entries_filtered) + else: + self.dir_entries_filtered_sorted = sorted( + entries_filtered, key=cmp_to_key(self._sort_cmp)) + i_row, i_col = 0, 0 + for i, item in enumerate(self.dir_entries_filtered_sorted): + if self.per_row == i_col: + i_col = 0 + i_row += 1 + slot = GallerySlot(item, item_clicker(i)) + self._grid.attach(slot, i_col, i_row, 1, 1) + self.slots += [slot] + i_col += 1 self._on_grid_built() self.selected_idx = 0 self._update_view() @@ -330,7 +390,7 @@ class Gallery: """Return how many diff. values for sort_attr in (un-)filtered store""" diversities = [0, 0] for i, store in enumerate([self.dir_entries, - [s.item for s in self.slots]]): + self.dir_entries_filtered_sorted]): values = set() for item in store: if isinstance(item, ImgItem): @@ -354,7 +414,7 @@ class Gallery: return -1 if isinstance(b, DirItem) and not isinstance(a, DirItem): return +1 - # apply ._sort_order within DirItems and FileItems (separately) + # apply ._sort_order within DirItems and ImgItems (separately) ret = 0 for key in [sorter.name for sorter in self._sort_order]: a_cmp = None @@ -455,32 +515,39 @@ class MainWindow(Gtk.Window): sort_store: Gtk.ListStore sort_selection: Gtk.SingleSelection prev_key: list - button_activate_sort: Gtk.Button counter: Gtk.Label + btn_activate_sort: Gtk.Button + btn_dec_per_row: Gtk.Button + btn_inc_per_row: Gtk.Button + btn_show_dirs: Gtk.Button def __init__(self, app, **kwargs): super().__init__(**kwargs) self.app = app def init_navbar(): - def add_button(label_, on_click, parent_box): - btn = Gtk.Button(label=label_) - btn.connect('clicked', on_click) - parent_box.append(btn) + + def add_button(label_, on_click, checkbox=False): + btn = (Gtk.CheckButton(label=label_) if checkbox + else Gtk.Button(label=label_)) + btn.connect('toggled' if checkbox else 'clicked', on_click) + navbar.append(btn) + return btn + navbar = Gtk.Box(orientation=OR_H) self.counter = Gtk.Label() navbar.append(self.counter) - add_button('sidebar', lambda _: self.toggle_side_box(), navbar) - add_button('reload', lambda _: self.load_directory(), navbar) + add_button('sidebar', lambda _: self.toggle_side_box()) + add_button('reload', lambda _: self.load_directory()) + self.btn_show_dirs = add_button('show directories', + self.reset_show_dirs, True) + add_button('recurse directories ', self.reset_recurse, True) navbar.append(Gtk.Label(label=' per row: ')) - add_button('less', lambda _: self.inc_per_row(-1), navbar) - add_button('more', lambda _: self.inc_per_row(+1), navbar) - btn = Gtk.CheckButton(label='show directories') - btn.connect('toggled', self.reset_show_dirs) - navbar.append(btn) - btn = Gtk.CheckButton(label='recurse directories') - btn.connect('toggled', self.reset_recurse) - navbar.append(btn) + add_button('by first sorter', self.reset_row_by_first_sorter, True) + self.btn_dec_per_row = add_button('less', + lambda _: self.inc_per_row(-1)) + self.btn_inc_per_row = add_button('more', + lambda _: self.inc_per_row(+1)) return navbar def init_metadata_box(): @@ -542,11 +609,11 @@ class MainWindow(Gtk.Window): sort_box = Gtk.Box(orientation=OR_V) sort_box.append(Gtk.Label(label='** sort order **')) sort_box.append(selector) - self.button_activate_sort = Gtk.Button(label='activate') - self.button_activate_sort.props.sensitive = False - self.button_activate_sort.connect( + self.btn_activate_sort = Gtk.Button(label='activate') + self.btn_activate_sort.props.sensitive = False + self.btn_activate_sort.connect( 'clicked', lambda _: self.activate_sort_order()) - sort_box.append(self.button_activate_sort) + sort_box.append(self.btn_activate_sort) return sort_box def init_key_control(): @@ -658,7 +725,7 @@ class MainWindow(Gtk.Window): sorter = self.sort_store.get_item(i) sorter.list_item.remove_css_class('temp') self.app.sort_order += [sorter] - self.button_activate_sort.props.sensitive = False + self.btn_activate_sort.props.sensitive = False old_selection = self.gallery.selected_item self.gallery.build_and_show(old_selection) @@ -715,6 +782,17 @@ class MainWindow(Gtk.Window): side_box_width = self.side_box.measure(OR_H, -1).natural self.gallery.on_resize(self.get_width() - side_box_width) + def reset_row_by_first_sorter(self, button): + """By button's .active, (un-)mirror top sorter in each gallery row.""" + self.gallery.per_row_by_first_sorter = button.props.active + self.btn_inc_per_row.set_sensitive(not button.props.active) + self.btn_dec_per_row.set_sensitive(not button.props.active) + self.btn_show_dirs.set_sensitive(not button.props.active) + if button.props.active: + self.btn_show_dirs.set_active(False) + self.btn_show_dirs.set_sensitive(not button.props.active) + self.gallery.build_and_show(self.gallery.selected_item) + def reset_show_dirs(self, button): """By button's .active, in-/exclude directories from gallery view.""" self.gallery.show_dirs = button.props.active @@ -755,7 +833,7 @@ class MainWindow(Gtk.Window): for i in range(self.sort_store.get_n_items()): sort_item = self.sort_store.get_item(i) sort_item.list_item.add_css_class('temp') - self.button_activate_sort.props.sensitive = True + self.btn_activate_sort.props.sensitive = True def move_selection_in_sort_order(self, direction): """Move sort order selection by direction (-1 or +1).""" @@ -777,7 +855,7 @@ class MainWindow(Gtk.Window): bookmarked = 'BOOKMARK' if selected_item.bookmarked else '' self.metadata.set_text( '\n'.join([title, bookmarked] + params_strs)) - total = len(self.gallery.slots) + total = len(self.gallery.dir_entries_filtered_sorted) self.counter.set_text(f' {self.gallery.selected_idx + 1} of {total} ') def handle_keypress(self, keyval): -- 2.30.2