1 #import <ObjXMPP/namespaces.h>
3 #include <gdk-pixbuf/gdk-pixbuf.h>
5 #import "JubGtkRosterUI.h"
6 #import "JubGObjectMap.h"
7 #import "JubGtkChatUI.h"
8 #import "JubGtkHelper.h"
10 static void roster_row_activated(GtkTreeView *tree_view, GtkTreePath *path,
11 GtkTreeViewColumn *column, gpointer data)
13 OFAutoreleasePool *pool;
15 JubChatClient *client = data;
17 GtkTreeModel *tree_model;
21 tree_model = gtk_tree_view_get_model(tree_view);
22 gtk_tree_model_get_iter(tree_model, &row_iter, path);
23 gtk_tree_model_get(tree_model, &row_iter, 1, &jid_s, -1);
25 // This was a group row
28 pool = [OFAutoreleasePool new];
30 jid = [OFString stringWithUTF8String: jid_s];
31 contact = [client.contactManager.contacts objectForKey: jid];
33 [client performSelectorOnMainThread: @selector(chatForContact:)
39 static void presence_changed(GtkComboBox *combo_box, gpointer data)
41 JubChatClient *client = data;
42 OFAutoreleasePool *pool = [OFAutoreleasePool new];
43 const char *status = gtk_combo_box_get_active_id(combo_box);
45 [client sendPresenceWithStatus: @(status)];
50 static gboolean filter_roster_by_presence(GtkTreeModel *model,
51 GtkTreeIter *iter, gpointer data)
54 gtk_tree_model_get(model, iter, 2, &status, -1);
56 // Groups have no status
60 if (!strcmp(status, "unavailable"))
66 static void dialog_response_callback(GtkDialog *dialog, gint response_id,
69 void (^block)(gint) = user_data;
74 @implementation JubGtkRosterUI
75 - initWithClient: (JubChatClient*)client
80 GtkTreeView *roster_view;
83 _groupMap = [[OFMapTable alloc]
84 initWithKeyFunctions: keyFunctions
85 valueFunctions: rowRefFunctions];
86 _subDialogMap = [[OFMapTable alloc]
87 initWithKeyFunctions: keyFunctions
88 valueFunctions: gObjectFunctions];
89 _contactMap = [[OFMutableDictionary alloc] init];
90 _client = [client retain];
92 [_client.contactManager addDelegate: self];
93 [_client.avatarManager setDelegate: self];
95 builder = gtk_builder_new();
96 gtk_builder_add_from_file(builder, "data/gtk/roster.ui", NULL);
97 gtk_builder_connect_signals(builder, NULL);
100 GTK_WIDGET(gtk_builder_get_object(builder, "RosterWindow"));
102 gtk_widget_show(_roster_window);
104 _roster_model = GTK_TREE_STORE(gtk_builder_get_object(builder,
107 _roster_filter = GTK_TREE_MODEL_FILTER(
108 gtk_builder_get_object(builder, "RosterTreeModelFilter"));
110 gtk_tree_model_filter_set_visible_func(_roster_filter,
111 filter_roster_by_presence, NULL, NULL);
113 roster_view = GTK_TREE_VIEW(gtk_builder_get_object(builder,
116 g_signal_connect(roster_view, "row_activated",
117 G_CALLBACK(roster_row_activated), client);
119 _presence_combo = GTK_COMBO_BOX(gtk_builder_get_object(builder,
120 "PresenceComboBox"));
122 _presence_combo_changed_handler_id =
123 g_signal_connect(_presence_combo, "changed",
124 G_CALLBACK(presence_changed), client);
126 g_object_unref(G_OBJECT(builder));
137 [_client.avatarManager setDelegate: nil];
138 [_client.contactManager removeDelegate: self];
139 [_subDialogMap release];
141 [_contactMap release];
144 gtk_widget_destroy(_roster_window);
149 - (void)Jub_closeSubscribeDialogForRosterItem: (XMPPRosterItem*)rosterItem
151 // Close subscripton dialogs, answered on another client
153 OFString *subscription = rosterItem.subscription;
154 OFString *bareJID = [rosterItem.JID bareJID];
156 if (([subscription isEqual: @"from"] ||
157 [subscription isEqual: @"both"]) &&
158 (dialog = [_subDialogMap valueForKey: bareJID]) != NULL)
159 gtk_dialog_response(dialog, GTK_RESPONSE_NONE);
162 /* Roster Delegate methods */
163 - (void)Jub_addRosterItem: (XMPPRosterItem*)item
164 group: (OFString*)group
166 GtkTreeIter group_iter, contact_iter;
167 GtkTreeRowReference *group_ref, *contact_ref;
168 GtkTreePath *group_path, *contact_path;
169 OFString *bareJID = [item.JID bareJID];
170 OFMapTable *contactRows;
172 if (!(contactRows = [_contactMap objectForKey: bareJID])) {
173 contactRows = [OFMapTable
174 mapTableWithKeyFunctions: keyFunctions
175 valueFunctions: rowRefFunctions];
177 [_contactMap setObject: contactRows
181 group_ref = [_groupMap valueForKey: group];
184 // Create new group row
185 gtk_tree_store_append(_roster_model, &group_iter, NULL);
186 gtk_tree_store_set(_roster_model, &group_iter,
187 0, [group UTF8String], -1);
189 group_path = gtk_tree_model_get_path(GTK_TREE_MODEL(
190 _roster_model), &group_iter);
192 group_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(
193 _roster_model), group_path);
195 [_groupMap setValue: group_ref
198 // Get iter for existing group row
199 group_path = gtk_tree_row_reference_get_path(group_ref);
201 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model),
202 &group_iter, group_path);
204 gtk_tree_path_free(group_path);
206 // Create new contact row
207 gtk_tree_store_append(_roster_model, &contact_iter, &group_iter);
209 gtk_tree_store_set(_roster_model, &contact_iter,
210 0, [item.name UTF8String],
211 1, [bareJID UTF8String],
212 2, "unavailable", -1);
213 else if (item.JID.node)
214 gtk_tree_store_set(_roster_model, &contact_iter,
215 0, [item.JID.node UTF8String],
216 1, [bareJID UTF8String],
217 2, "unavailable", -1);
219 gtk_tree_store_set(_roster_model, &contact_iter,
220 0, [item.JID.domain UTF8String],
221 1, [bareJID UTF8String],
222 2, "unavailable", -1);
224 contact_path = gtk_tree_model_get_path(GTK_TREE_MODEL(_roster_model),
227 contact_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(_roster_model),
230 gtk_tree_path_free(contact_path);
232 [contactRows setValue: contact_ref
236 - (void)Jub_removeRosterItem: (XMPPRosterItem*)item
237 group: (OFString*)group
239 GtkTreeIter contact_iter, group_iter;
240 GtkTreePath *contact_path, *group_path;
241 GtkTreeRowReference *contact_ref, *group_ref;
242 OFString *bareJID = [item.JID bareJID];
243 OFMapTable *contactRows = [_contactMap objectForKey: bareJID];
245 contact_ref = [contactRows valueForKey: group];
246 contact_path = gtk_tree_row_reference_get_path(contact_ref);
247 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model), &contact_iter,
249 gtk_tree_path_free(contact_path);
251 gtk_tree_store_remove(_roster_model, &contact_iter);
253 group_ref = [_groupMap valueForKey: group];
254 group_path = gtk_tree_row_reference_get_path(group_ref);
255 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model), &group_iter,
258 if (!gtk_tree_model_iter_has_child(GTK_TREE_MODEL(_roster_model),
260 gtk_tree_store_remove(_roster_model, &group_iter);
261 [_groupMap removeValueForKey: group];
264 gtk_tree_path_free(group_path);
266 [contactRows removeValueForKey: group];
267 if([contactRows count] == 0)
268 [_contactMap removeObjectForKey: bareJID];
271 - (void)contactManager: (XMPPContactManager*)manager
272 didAddContact: (XMPPContact*)contact
274 XMPPRosterItem *rosterItem = contact.rosterItem;
275 OFArray *groups = rosterItem.groups;;
278 groups = @[ @"General" ];
280 for (OFString *group in groups)
281 [self performSelectorOnGLibThread: @selector(
282 Jub_addRosterItem:group:)
283 withObject: rosterItem
286 [self Jub_closeSubscribeDialogForRosterItem: rosterItem];
289 - (void)contactManager: (XMPPContactManager*)manager
290 didRemoveContact: (XMPPContact*)contact
292 XMPPRosterItem *rosterItem = contact.rosterItem;
293 OFArray *groups = rosterItem.groups;
296 groups = @[ @"General" ];
298 for (OFString *group in groups)
299 [self performSelectorOnGLibThread: @selector(
300 Jub_removeRosterItem:group:)
301 withObject: rosterItem
305 - (void)contactManager: (XMPPContactManager*)manager
306 didReceiveSubscriptionRequest: (XMPPPresence*)presence
308 XMPPJID *JID = presence.from;
309 OFString *bareJID = [JID bareJID];
310 OFString *message = [OFString stringWithFormat: @"<b>%@</b> would like "
311 @"to subscribe to your presence.", bareJID];
313 GtkWidget *dialog, *content_area, *label;
315 if ([_subDialogMap valueForKey: JID] != NULL)
318 dialog = gtk_dialog_new_with_buttons("Subscription Request",
319 GTK_WINDOW(_roster_window), GTK_DIALOG_DESTROY_WITH_PARENT,
320 "Accept", GTK_RESPONSE_ACCEPT,
321 "Deny", GTK_RESPONSE_REJECT, NULL);
323 content_area = gtk_dialog_get_content_area(GTK_DIALOG(dialog));
324 label = gtk_label_new(NULL);
325 gtk_label_set_markup(GTK_LABEL(label), [message UTF8String]);
326 gtk_container_add(GTK_CONTAINER(content_area), label);
328 g_signal_connect(dialog, "response",
329 G_CALLBACK(dialog_response_callback),
330 [^(gint response_id) {
331 if (response_id == GTK_RESPONSE_ACCEPT)
332 [manager sendSubscribedToJID: JID];
333 else if (response_id == GTK_RESPONSE_REJECT)
334 [manager sendUnsubscribedToJID: JID];
335 [_subDialogMap removeValueForKey: bareJID];
336 gtk_widget_destroy(GTK_WIDGET(dialog));
339 [_subDialogMap setValue: dialog
342 gtk_widget_show_all(dialog);
346 - (void)contact: (XMPPContact*)contact
347 willUpdateWithRosterItem: (XMPPRosterItem*)rosterItem;
349 [self Jub_closeSubscribeDialogForRosterItem: rosterItem];
351 // Remove contact from old set of groups
352 XMPPRosterItem *oldItem = contact.rosterItem;
353 OFArray *groups = oldItem.groups;
356 groups = @[ @"General" ];
358 for (OFString *group in groups)
359 [self performSelectorOnGLibThread: @selector(
360 Jub_removeRosterItem:group:)
364 // Add contact to new set of groups
365 groups = rosterItem.groups;
368 groups = @[ @"General" ];
370 for (OFString *group in groups)
371 [self performSelectorOnGLibThread: @selector(
372 Jub_addRosterItem:group:)
373 withObject: rosterItem
377 - (void)contact: (XMPPContact*)contact
378 didSendPresence: (XMPPPresence*)presence
380 OFDictionary *allPresences = [contact presences];
381 XMPPPresence *highPresence = [[[allPresences allObjects] sortedArray]
383 OFMutableString *tooltip =
384 [OFMutableString stringWithString: @"<b>Resources:</b>"];
386 [allPresences enumerateKeysAndObjectsUsingBlock:
387 ^(OFString *resource, XMPPPresence *pres, bool *stop) {
388 [tooltip appendString: @"\n"];
389 [tooltip appendString: resource];
390 if ([pres.type isEqual: @"available"]) {
391 if (pres.show != nil)
392 [tooltip appendFormat: @" (%@)", pres.show];
394 [tooltip appendString: @" (available)"];
396 [tooltip appendString: @" (unavailable)"];
399 [tooltip appendFormat: @": <i>%@</i>", pres.status];
405 GtkTreeRowReference *ref;
406 OFString *bareJID = [contact.rosterItem.JID bareJID];
407 OFMapTable *contactRows = [_contactMap objectForKey: bareJID];
408 OFArray *groups = contact.rosterItem.groups;;
411 groups = @[ @"General" ];
413 for (OFString *group in groups) {
414 ref = [contactRows valueForKey: group];
415 path = gtk_tree_row_reference_get_path(ref);
416 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model),
418 gtk_tree_path_free(path);
420 if ([highPresence.type isEqual: @"available"]) {
421 if (highPresence.show != nil)
422 gtk_tree_store_set(_roster_model, &iter,
423 2, [highPresence.show UTF8String],
426 gtk_tree_store_set(_roster_model, &iter,
429 gtk_tree_store_set(_roster_model, &iter,
430 2, "unavailable", -1);
432 gtk_tree_store_set(_roster_model, &iter,
433 3, [tooltip UTF8String], -1);
436 gtk_tree_model_filter_refilter(_roster_filter);
440 - (void)contact: (XMPPContact*)contact
441 didSetAvatar: (OFString*)avatarFile
443 of_log(@"Got an avatar from %@", contact.rosterItem.JID);
447 GtkTreeRowReference *ref;
448 OFString *bareJID = [contact.rosterItem.JID bareJID];
449 OFMapTable *contactRows = [_contactMap objectForKey: bareJID];
450 OFArray *groups = contact.rosterItem.groups;;
453 gdk_pixbuf_new_from_file([avatarFile UTF8String], NULL);
456 groups = @[ @"General" ];
458 for (OFString *group in groups) {
459 ref = [contactRows valueForKey: group];
460 path = gtk_tree_row_reference_get_path(ref);
461 gtk_tree_model_get_iter(GTK_TREE_MODEL(_roster_model),
463 gtk_tree_path_free(path);
465 gtk_tree_store_set(_roster_model, &iter,
468 g_object_unref(G_OBJECT(avatar));
472 - (void)client: (JubChatClient*)client_
473 didChangePresence: (XMPPPresence*)presence
475 OFString *tooltip = @"";
477 [[presence elementForName: @"show"
478 namespace: XMPP_NS_CLIENT] stringValue];
480 [[presence elementForName: @"status"
481 namespace: XMPP_NS_CLIENT] stringValue];
484 tooltip = [@"<b>Status:</b> " stringByAppendingString: status];
487 // Block the PresenceComboBox's changed handler, so it doesn't
488 // fire and resend presence
489 g_signal_handler_block(_presence_combo,
490 _presence_combo_changed_handler_id);
492 if ([presence.type isEqual: @"unavailable"])
493 gtk_combo_box_set_active_id(_presence_combo,
495 else if (show == nil)
496 gtk_combo_box_set_active_id(_presence_combo,
499 gtk_combo_box_set_active_id(_presence_combo,
502 // Unblock the changed handler
503 g_signal_handler_unblock(_presence_combo,
504 _presence_combo_changed_handler_id);
506 gtk_widget_set_tooltip_markup(GTK_WIDGET(_presence_combo),
507 [tooltip UTF8String]);