1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
|
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<Jid, Widget> rows = new HashMap<Jid, Widget>(Jid.hash_func, Jid.equals_func);
public HashMap<Widget, ListRow> row_wrappers = new HashMap<Widget, ListRow>();
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<Jid>? 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 };
title_label.set_markup(@"<b>$(Markup.escape_text(aff_str))</b>");
Label count_label = new Label(@"$count") { xalign=0, margin_end=7, hexpand=true };
count_label.add_css_class("dim-label");
Grid grid = new Grid() { margin_top=top?5:15, column_spacing=5, hexpand=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 }, 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);
}
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;
}
}
}
}
|