using Gee; using Gtk; using Dino.Entities; using Xmpp; namespace Dino.Ui.OccupantMenu { [GtkTemplate (ui = "/im/dino/Dino/occupant_list.ui")] public class List : Box { public signal void conversation_selected(Conversation? conversation); private StreamInteractor stream_interactor; [GtkChild] public unowned ListBox list_box; [GtkChild] private unowned SearchEntry search_entry; private Conversation conversation; private string[]? filter_values; private HashMap rows = new HashMap(Jid.hash_func, Jid.equals_func); public HashMap row_wrappers = new HashMap(); public List(StreamInteractor stream_interactor, Conversation conversation) { this.stream_interactor = stream_interactor; list_box.set_header_func(header); list_box.set_sort_func(sort); list_box.set_filter_func(filter); search_entry.search_changed.connect(refilter); stream_interactor.get_module(PresenceManager.IDENTITY).show_received.connect(on_show_received); stream_interactor.get_module(PresenceManager.IDENTITY).received_offline_presence.connect(on_received_offline_presence); initialize_for_conversation(conversation); } public void initialize_for_conversation(Conversation conversation) { this.conversation = conversation; Gee.List? occupants = stream_interactor.get_module(MucManager.IDENTITY).get_occupants(conversation.counterpart, conversation.account); if (occupants != null) { foreach (Jid occupant in occupants) { add_occupant(occupant); } } list_box.invalidate_filter(); } private void refilter() { string[]? values = null; string str = search_entry.get_text (); if (str != "") values = str.split(" "); if (filter_values == values) return; filter_values = values; list_box.invalidate_filter(); } public void add_occupant(Jid jid) { var row_wrapper = new ListRow(stream_interactor, conversation, jid); var widget = row_wrapper.get_widget(); row_wrappers[widget] = row_wrapper; rows[jid] = widget; list_box.append(widget); } public void remove_occupant(Jid jid) { list_box.remove(rows[jid]); rows.unset(jid); } private void on_received_offline_presence(Jid jid, Account account) { if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) { if (rows.has_key(jid)) { remove_occupant(jid); } list_box.invalidate_filter(); } } private void on_show_received(Jid jid, Account account) { if (conversation != null && conversation.counterpart.equals_bare(jid) && jid.is_full()) { if (!rows.has_key(jid)) { add_occupant(jid); } list_box.invalidate_filter(); } } private void header(ListBoxRow row, ListBoxRow? before_row) { ListRow row_wrapper1 = row_wrappers[row.get_child()]; Xmpp.Xep.Muc.Affiliation? a1 = stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper1.jid, row_wrapper1.conversation.account); if (a1 == null) return; if (before_row != null) { ListRow row_wrapper2 = row_wrappers[before_row.get_child()]; Xmpp.Xep.Muc.Affiliation? a2 = stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper2.jid, row_wrapper2.conversation.account); if (a1 != a2) { row.set_header(generate_header_widget(a1, false)); } else if (row.get_header() != null){ row.set_header(null); } } else { row.set_header(generate_header_widget(a1, true)); } } private Widget generate_header_widget(Xmpp.Xep.Muc.Affiliation affiliation, bool top) { string aff_str; switch (affiliation) { case Xmpp.Xep.Muc.Affiliation.OWNER: aff_str = _("Owner"); break; case Xmpp.Xep.Muc.Affiliation.ADMIN: aff_str = _("Admin"); break; case Xmpp.Xep.Muc.Affiliation.MEMBER: aff_str = _("Member"); break; default: aff_str = _("User"); break; } int count = 0; foreach (ListRow row in row_wrappers.values) { Xmpp.Xep.Muc.Affiliation aff = stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row.jid, conversation.account); if (aff == affiliation) count++; } Label title_label = new Label("") { margin_start=10, xalign=0, visible=true }; title_label.set_markup(@"$(Markup.escape_text(aff_str))"); Label count_label = new Label(@"$count") { xalign=0, margin_end=7, hexpand=true, visible=true }; count_label.get_style_context().add_class("dim-label"); Grid grid = new Grid() { margin_top=top?5:15, column_spacing=5, hexpand=true, visible=true }; grid.attach(title_label, 0, 0, 1, 1); grid.attach(count_label, 1, 0, 1, 1); grid.attach(new Separator(Orientation.HORIZONTAL) { hexpand=true, vexpand=true, visible=true }, 0, 1, 2, 1); return grid; } private bool filter(ListBoxRow r) { ListRow row_wrapper = row_wrappers[r.get_child()]; foreach (string filter in filter_values) { return row_wrapper.name_label.label.down().contains(filter.down()); } return true; } private int sort(ListBoxRow row1, ListBoxRow row2) { ListRow row_wrapper1 = row_wrappers[row1.get_child()]; ListRow row_wrapper2 = row_wrappers[row2.get_child()]; int affiliation1 = get_affiliation_ranking(stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper1.jid, row_wrapper1.conversation.account) ?? Xmpp.Xep.Muc.Affiliation.NONE); int affiliation2 = get_affiliation_ranking(stream_interactor.get_module(MucManager.IDENTITY).get_affiliation(conversation.counterpart, row_wrapper2.jid, row_wrapper2.conversation.account) ?? Xmpp.Xep.Muc.Affiliation.NONE); if (affiliation1 < affiliation2) return -1; else if (affiliation1 > affiliation2) return 1; else return row_wrapper1.name_label.label.collate(row_wrapper2.name_label.label); return 0; } private int get_affiliation_ranking(Xmpp.Xep.Muc.Affiliation affiliation) { switch (affiliation) { case Xmpp.Xep.Muc.Affiliation.OWNER: return 1; case Xmpp.Xep.Muc.Affiliation.ADMIN: return 2; case Xmpp.Xep.Muc.Affiliation.MEMBER: return 3; default: return 4; } } } }