1 #import <ObjXMPP/namespaces.h>
4 #import "JubGtkRosterUI.h"
5 #import "JubGObjectMap.h"
6 #import "JubGtkChatUI.h"
7 #import "JubGtkHelper.h"
9 static void roster_row_activated(GtkTreeView *tree_view, GtkTreePath *path,
10 GtkTreeViewColumn *column, gpointer data)
12 OFAutoreleasePool *pool;
14 JubChatClient *client = data;
16 GtkTreeModel *tree_model;
20 tree_model = gtk_tree_view_get_model(tree_view);
21 gtk_tree_model_get_iter(tree_model, &row_iter, path);
22 gtk_tree_model_get(tree_model, &row_iter, 1, &jid_s, -1);
24 // This was a group row
27 pool = [OFAutoreleasePool new];
29 jid = [OFString stringWithUTF8String: jid_s];
30 contact = [client.contactManager.contacts objectForKey: jid];
32 [client performSelectorOnMainThread: @selector(chatForContact:)
38 static void presence_changed(GtkComboBox *combo_box, gpointer data)
41 XMPPConnection *connection = data;
42 OFAutoreleasePool *pool = [OFAutoreleasePool new];
44 const char *status = gtk_combo_box_get_active_id(combo_box);
46 if (!strcmp(status, "unavailable"))
47 pres = [XMPPPresence presenceWithType: @"unavailable"];
49 pres = [XMPPPresence presence];
50 if (strcmp(status, "available"))
51 [pres setShow: @(status)];
54 [connection sendStanza: pres];
59 static gboolean filter_roster_by_presence(GtkTreeModel *model,
60 GtkTreeIter *iter, gpointer data)
63 gtk_tree_model_get(model, iter, 2, &status, -1);
65 // Groups have no status
69 if (!strcmp(status, "unavailable"))
75 @implementation JubGtkRosterUI
76 - initWithClient: (JubChatClient*)client
81 GtkTreeView *roster_view;
84 _groupMap = [[OFMapTable alloc]
85 initWithKeyFunctions: keyFunctions
86 valueFunctions: rowRefFunctions];
87 _contactMap = [[OFMutableDictionary alloc] init];
88 _client = [client retain];
90 [_client.contactManager addDelegate: self];
92 builder = gtk_builder_new();
93 gtk_builder_add_from_file(builder, "data/gtk/roster.ui", NULL);
94 gtk_builder_connect_signals(builder, NULL);
97 GTK_WIDGET(gtk_builder_get_object(builder, "RosterWindow"));
99 gtk_widget_show(_roster_window);
101 _roster_model = GTK_TREE_STORE(gtk_builder_get_object(builder,
104 _roster_filter = GTK_TREE_MODEL_FILTER(
105 gtk_builder_get_object(builder, "RosterTreeModelFilter"));
107 gtk_tree_model_filter_set_visible_func(_roster_filter,
108 filter_roster_by_presence, NULL, NULL);
110 roster_view = GTK_TREE_VIEW(gtk_builder_get_object(builder,
113 g_signal_connect(roster_view, "row_activated",
114 G_CALLBACK(roster_row_activated), client);
116 _presence_combo = GTK_COMBO_BOX(gtk_builder_get_object(builder,
117 "PresenceComboBox"));
119 _presence_combo_changed_handler_id =
120 g_signal_connect(_presence_combo, "changed",
121 G_CALLBACK(presence_changed), client.connection);
123 g_object_unref(G_OBJECT(builder));
134 [_client.contactManager removeDelegate: self];
136 [_contactMap release];
139 gtk_widget_destroy(_roster_window);
144 /* Roster Delegate methods */
145 - (void)Jub_addRosterItem: (XMPPRosterItem*)item
146 group: (OFString*)group
148 GtkTreeIter group_iter, contact_iter;
149 GtkTreeRowReference *group_ref, *contact_ref;
150 GtkTreePath *group_path, *contact_path;
151 OFString *bareJID = [item.JID bareJID];
152 OFMapTable *contactRows;
154 if (!(contactRows = [_contactMap objectForKey: bareJID])) {
155 contactRows = [OFMapTable
156 mapTableWithKeyFunctions: keyFunctions
157 valueFunctions: rowRefFunctions];
159 [_contactMap setObject: contactRows
163 group_ref = [_groupMap valueForKey: group];
166 // Create new group row
167 gtk_tree_store_append(_roster_model, &group_iter, NULL);
168 gtk_tree_store_set(_roster_model, &group_iter,
169 0, [group UTF8String], -1);
171 group_path = gtk_tree_model_get_path(GTK_TREE_MODEL(
172 _roster_model), &group_iter);
174 group_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(
175 _roster_model), group_path);
177 [_groupMap setValue: group_ref
180 // Get iter for existing group row
181 group_path = gtk_tree_row_reference_get_path(group_ref);
183 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model),
184 &group_iter, group_path);
186 gtk_tree_path_free(group_path);
188 // Create new contact row
189 gtk_tree_store_append(_roster_model, &contact_iter, &group_iter);
191 gtk_tree_store_set(_roster_model, &contact_iter,
192 0, [item.name UTF8String],
193 1, [bareJID UTF8String],
194 2, "unavailable", -1);
195 else if (item.JID.node)
196 gtk_tree_store_set(_roster_model, &contact_iter,
197 0, [item.JID.node UTF8String],
198 1, [bareJID UTF8String],
199 2, "unavailable", -1);
201 gtk_tree_store_set(_roster_model, &contact_iter,
202 0, [item.JID.domain UTF8String],
203 1, [bareJID UTF8String],
204 2, "unavailable", -1);
206 contact_path = gtk_tree_model_get_path(GTK_TREE_MODEL(_roster_model),
209 contact_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(_roster_model),
212 gtk_tree_path_free(contact_path);
214 [contactRows setValue: contact_ref
218 - (void)Jub_removeRosterItem: (XMPPRosterItem*)item
219 group: (OFString*)group
221 GtkTreeIter contact_iter, group_iter;
222 GtkTreePath *contact_path, *group_path;
223 GtkTreeRowReference *contact_ref, *group_ref;
224 OFString *bareJID = [item.JID bareJID];
225 OFMapTable *contactRows = [_contactMap objectForKey: bareJID];
227 contact_ref = [contactRows valueForKey: group];
228 contact_path = gtk_tree_row_reference_get_path(contact_ref);
229 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model), &contact_iter,
231 gtk_tree_path_free(contact_path);
233 gtk_tree_store_remove(_roster_model, &contact_iter);
235 group_ref = [_groupMap valueForKey: group];
236 group_path = gtk_tree_row_reference_get_path(group_ref);
237 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model), &group_iter,
240 if (!gtk_tree_model_iter_has_child(GTK_TREE_MODEL(_roster_model),
242 gtk_tree_store_remove(_roster_model, &group_iter);
243 [_groupMap removeValueForKey: group];
246 gtk_tree_path_free(group_path);
248 [contactRows removeValueForKey: group];
249 if([contactRows count] == 0)
250 [_contactMap removeObjectForKey: bareJID];
253 - (void)contactManager: (XMPPContactManager*)manager
254 didAddContact: (XMPPContact*)contact
256 XMPPRosterItem *rosterItem = contact.rosterItem;
257 OFArray *groups = rosterItem.groups;;
260 groups = @[ @"General" ];
262 for (OFString *group in groups)
263 [self performSelectorOnGLibThread: @selector(
264 Jub_addRosterItem:group:)
265 withObject: rosterItem
269 - (void)contactManager: (XMPPContactManager*)manager
270 didRemoveContact: (XMPPContact*)contact
272 XMPPRosterItem *rosterItem = contact.rosterItem;
273 OFArray *groups = rosterItem.groups;
276 groups = @[ @"General" ];
278 for (OFString *group in groups)
279 [self performSelectorOnGLibThread: @selector(
280 Jub_removeRosterItem:group:)
281 withObject: rosterItem
285 - (void)contact: (XMPPContact*)contact
286 willUpdateWithRosterItem: (XMPPRosterItem*)rosterItem;
288 // Remove contact from old set of groups
289 XMPPRosterItem *oldItem = contact.rosterItem;
290 OFArray *groups = oldItem.groups;
293 groups = @[ @"General" ];
295 for (OFString *group in groups)
296 [self performSelectorOnGLibThread: @selector(
297 Jub_removeRosterItem:group:)
301 // Add contact to new set of groups
302 groups = rosterItem.groups;
305 groups = @[ @"General" ];
307 for (OFString *group in groups)
308 [self performSelectorOnGLibThread: @selector(
309 Jub_addRosterItem:group:)
310 withObject: rosterItem
314 - (void)contact: (XMPPContact*)contact
315 didSendPresence: (XMPPPresence*)presence
317 OFDictionary *allPresences = [contact presences];
318 XMPPPresence *highPresence = [[[allPresences allObjects] sortedArray]
320 OFMutableString *tooltip =
321 [OFMutableString stringWithString: @"<b>Resources:</b>"];
323 [allPresences enumerateKeysAndObjectsUsingBlock:
324 ^(OFString *resource, XMPPPresence *pres, BOOL *stop) {
325 [tooltip appendString: @"\n"];
326 [tooltip appendString: resource];
327 if ([pres.type isEqual: @"available"]) {
328 if (pres.show != nil)
329 [tooltip appendFormat: @" (%@)", pres.show];
331 [tooltip appendString: @" (available)"];
333 [tooltip appendString: @" (unavailable)"];
336 [tooltip appendFormat: @": <i>%@</i>", pres.status];
342 GtkTreeRowReference *ref;
343 OFString *bareJID = [contact.rosterItem.JID bareJID];
344 OFMapTable *contactRows = [_contactMap objectForKey: bareJID];
345 OFArray *groups = contact.rosterItem.groups;;
348 groups = @[ @"General" ];
350 for (OFString *group in groups) {
351 ref = [contactRows valueForKey: group];
352 path = gtk_tree_row_reference_get_path(ref);
353 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model),
355 gtk_tree_path_free(path);
357 if ([highPresence.type isEqual: @"available"]) {
358 if (highPresence.show != nil)
359 gtk_tree_store_set(_roster_model, &iter,
360 2, [highPresence.show UTF8String],
363 gtk_tree_store_set(_roster_model, &iter,
366 gtk_tree_store_set(_roster_model, &iter,
367 2, "unavailable", -1);
369 gtk_tree_store_set(_roster_model, &iter,
370 3, [tooltip UTF8String], -1);
373 gtk_tree_model_filter_refilter(_roster_filter);
377 - (void)client: (JubChatClient*)client_
378 didChangePresence: (XMPPPresence*)presence
380 OFString *tooltip = @"";
382 [[presence elementForName: @"show"
383 namespace: XMPP_NS_CLIENT] stringValue];
385 [[presence elementForName: @"status"
386 namespace: XMPP_NS_CLIENT] stringValue];
389 tooltip = [@"<b>Status:</b> " stringByAppendingString: status];
392 // Block the PresenceComboBox's changed handler, so it doesn't
393 // fire and resend presence
394 g_signal_handler_block(_presence_combo,
395 _presence_combo_changed_handler_id);
397 if ([presence.type isEqual: @"unavailable"])
398 gtk_combo_box_set_active_id(_presence_combo,
400 else if (show == nil)
401 gtk_combo_box_set_active_id(_presence_combo,
404 gtk_combo_box_set_active_id(_presence_combo,
407 // Unblock the changed handler
408 g_signal_handler_unblock(_presence_combo,
409 _presence_combo_changed_handler_id);
411 gtk_widget_set_tooltip_markup(GTK_WIDGET(_presence_combo),
412 [tooltip UTF8String]);