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 JubGtkRosterUI *roster = data;
14 GtkTreeModel *tree_model;
17 OFAutoreleasePool *pool;
19 tree_model = gtk_tree_view_get_model(tree_view);
20 gtk_tree_model_get_iter(tree_model, &row_iter, path);
21 gtk_tree_model_get(tree_model, &row_iter, 1, &jid_s, -1);
23 // This was a group row
26 pool = [OFAutoreleasePool new];
27 jid = [XMPPJID JIDWithString: [OFString stringWithUTF8String: jid_s]];
29 [roster performSelectorOnMainThread: @selector(chatForJID:)
35 static void presence_changed(GtkComboBox *combo_box, gpointer data)
38 XMPPConnection *connection = data;
39 OFAutoreleasePool *pool = [OFAutoreleasePool new];
41 const char *status = gtk_combo_box_get_active_id(combo_box);
43 if (!strcmp(status, "unavailable"))
44 pres = [XMPPPresence presenceWithType: @"unavailable"];
46 pres = [XMPPPresence presence];
47 if (strcmp(status, "available"))
48 [pres setShow: @(status)];
51 [connection sendStanza: pres];
56 static gboolean filter_roster_by_presence(GtkTreeModel *model,
57 GtkTreeIter *iter, gpointer data)
60 gtk_tree_model_get(model, iter, 2, &status, -1);
62 // Groups have no status
66 if (!strcmp(status, "unavailable"))
72 @implementation JubGtkRosterUI
73 - initWithClient: (JubChatClient*)client_
78 GtkTreeView *roster_view;
81 groupMap = [[OFMapTable alloc]
82 initWithKeyFunctions: keyFunctions
83 valueFunctions: rowRefFunctions];
84 contactMap = [[OFMutableDictionary alloc] init];
85 chatMap = [[OFMutableDictionary alloc] init];
86 client = [client_ retain];
88 [client.contactManager addDelegate: self];
90 builder = gtk_builder_new();
91 gtk_builder_add_from_file(builder, "data/gtk/roster.ui", NULL);
92 gtk_builder_connect_signals(builder, NULL);
95 GTK_WIDGET(gtk_builder_get_object(builder, "RosterWindow"));
97 gtk_widget_show(roster_window);
99 roster_model = GTK_TREE_STORE(gtk_builder_get_object(builder,
102 roster_filter = GTK_TREE_MODEL_FILTER(
103 gtk_builder_get_object(builder, "RosterTreeModelFilter"));
105 gtk_tree_model_filter_set_visible_func(roster_filter,
106 filter_roster_by_presence, NULL, NULL);
108 roster_view = GTK_TREE_VIEW(gtk_builder_get_object(builder,
111 g_signal_connect(roster_view, "row_activated",
112 G_CALLBACK(roster_row_activated), self);
114 presence_combo = GTK_COMBO_BOX(gtk_builder_get_object(builder,
115 "PresenceComboBox"));
117 presence_combo_changed_handler_id =
118 g_signal_connect(presence_combo, "changed",
119 G_CALLBACK(presence_changed), client.connection);
121 g_object_unref(G_OBJECT(builder));
132 [client.contactManager removeDelegate: self];
134 [contactMap release];
138 gtk_widget_destroy(roster_window);
143 // FIXME: This needs to move somewhere else
144 - (JubGtkChatUI*)chatForJID: (XMPPJID*)jid
146 OFAutoreleasePool *pool = [OFAutoreleasePool new];
148 [chatMap objectForKey: [jid bareJID]];
150 OFString * title = [@"Chat with " stringByAppendingString:
153 chat = [[[JubGtkChatUI alloc]
156 [chatMap removeObjectForKey: [jid bareJID]];
158 sendBlock: ^(OFString *text) {
160 [XMPPMessage messageWithType: @"chat"];
163 [client.connection sendStanza: msg];
167 [chatMap setObject: chat
168 forKey: [jid bareJID]];
176 - (void)contact: (XMPPContact*)contact
177 didSendMessage: (XMPPMessage*)message
179 if (message.body == nil || ![message.type isEqual: @"chat"])
182 JubGtkChatUI *chat = [self chatForJID: message.from];
183 [chat addMessage: message.body
184 sender: [message.from bareJID]];
187 /* Roster Delegate methods */
188 - (void)Jub_addRosterItem: (XMPPRosterItem*)item
189 group: (OFString*)group
192 GtkTreeIter group_iter, contact_iter;
193 GtkTreeRowReference *group_ref, *contact_ref;
194 GtkTreePath *group_path, *contact_path;
195 OFString *bareJID = [item.JID bareJID];
196 OFMapTable *contactRows;
198 if (!(contactRows = [contactMap objectForKey: bareJID])) {
199 contactRows = [OFMapTable
200 mapTableWithKeyFunctions: keyFunctions
201 valueFunctions: rowRefFunctions];
203 [contactMap setObject: contactRows
207 group_ref = [groupMap valueForKey: group];
210 // Create new group row
211 gtk_tree_store_append(roster_model, &group_iter, NULL);
212 gtk_tree_store_set(roster_model, &group_iter,
213 0, [group UTF8String], -1);
215 group_path = gtk_tree_model_get_path(GTK_TREE_MODEL(
216 roster_model), &group_iter);
218 group_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(
219 roster_model), group_path);
221 [groupMap setValue: group_ref
224 // Get iter for existing group row
225 group_path = gtk_tree_row_reference_get_path(group_ref);
227 gtk_tree_model_get_iter(GTK_TREE_MODEL(roster_model),
228 &group_iter, group_path);
230 gtk_tree_path_free(group_path);
232 // Create new contact row
233 gtk_tree_store_append(roster_model, &contact_iter, &group_iter);
235 gtk_tree_store_set(roster_model, &contact_iter,
236 0, [item.name UTF8String],
237 1, [bareJID UTF8String],
238 2, "unavailable", -1);
240 gtk_tree_store_set(roster_model, &contact_iter,
241 0, [item.JID.node UTF8String],
242 1, [bareJID UTF8String],
243 2, "unavailable", -1);
245 contact_path = gtk_tree_model_get_path(GTK_TREE_MODEL(
246 roster_model), &contact_iter);
248 contact_ref = gtk_tree_row_reference_new(GTK_TREE_MODEL(
249 roster_model), contact_path);
251 gtk_tree_path_free(contact_path);
253 [contactRows setValue: contact_ref
258 - (void)Jub_removeRosterItem: (XMPPRosterItem*)item
259 group: (OFString*)group
262 GtkTreeIter contact_iter, group_iter;
263 GtkTreePath *contact_path, *group_path;
264 GtkTreeRowReference *contact_ref, *group_ref;
265 OFString *bareJID = [item.JID bareJID];
266 OFMapTable *contactRows = [contactMap objectForKey: bareJID];
268 contact_ref = [contactRows valueForKey: group];
269 contact_path = gtk_tree_row_reference_get_path(contact_ref);
270 gtk_tree_model_get_iter(GTK_TREE_MODEL(roster_model),
271 &contact_iter, contact_path);
272 gtk_tree_path_free(contact_path);
274 gtk_tree_store_remove(roster_model, &contact_iter);
276 group_ref = [groupMap valueForKey: group];
277 group_path = gtk_tree_row_reference_get_path(group_ref);
278 gtk_tree_model_get_iter(GTK_TREE_MODEL(roster_model),
279 &group_iter, group_path);
281 if (!gtk_tree_model_iter_has_child(GTK_TREE_MODEL(roster_model),
283 gtk_tree_store_remove(roster_model, &group_iter);
284 [groupMap removeValueForKey: group];
287 gtk_tree_path_free(group_path);
289 [contactRows removeValueForKey: group];
290 if([contactRows count] == 0)
291 [contactMap removeObjectForKey: bareJID];
295 - (void)contactManager: (XMPPContactManager*)manager
296 didAddContact: (XMPPContact*)contact
298 XMPPRosterItem *rosterItem = contact.rosterItem;
299 OFArray *groups = rosterItem.groups;;
302 groups = @[ @"General" ];
304 for (OFString *group in groups)
305 [self Jub_addRosterItem: rosterItem
309 - (void)contactManager: (XMPPContactManager*)manager
310 didRemoveContact: (XMPPContact*)contact
312 XMPPRosterItem *rosterItem = contact.rosterItem;
313 OFArray *groups = rosterItem.groups;
316 groups = @[ @"General" ];
318 for (OFString *group in groups)
319 [self Jub_removeRosterItem: rosterItem
323 - (void)contact: (XMPPContact*)contact
324 willUpdateWithRosterItem: (XMPPRosterItem*)rosterItem;
326 // Remove contact from old set of groups
327 XMPPRosterItem *oldItem = contact.rosterItem;
328 OFArray *groups = oldItem.groups;
331 groups = @[ @"General" ];
333 for (OFString *group in groups)
334 [self Jub_removeRosterItem: oldItem
337 // Add contact to new set of groups
338 groups = rosterItem.groups;
341 groups = @[ @"General" ];
343 for (OFString *group in groups)
344 [self Jub_addRosterItem: rosterItem
348 - (void)contact: (XMPPContact*)contact
349 didSendPresence: (XMPPPresence*)presence
354 GtkTreeRowReference *ref;
355 OFString *bareJID = [contact.rosterItem.JID bareJID];
356 OFMapTable *contactRows = [contactMap objectForKey: bareJID];
358 for (OFString *group in contact.rosterItem.groups) {
359 ref = [contactRows valueForKey: group];
360 path = gtk_tree_row_reference_get_path(ref);
361 gtk_tree_model_get_iter(GTK_TREE_MODEL(roster_model),
363 gtk_tree_path_free(path);
365 if ([presence.type isEqual: @"available"])
366 if (presence.show != nil)
367 gtk_tree_store_set(roster_model, &iter,
368 2, [presence.show UTF8String], -1);
370 gtk_tree_store_set(roster_model, &iter,
372 else if ([presence.type isEqual: @"unavailable"])
373 gtk_tree_store_set(roster_model, &iter,
374 2, "unavailable", -1);
376 gtk_tree_store_set(roster_model, &iter,
377 3, [presence.status UTF8String], -1);
380 gtk_tree_model_filter_refilter(roster_filter);
384 - (void)client: (JubChatClient*)client_
385 didChangePresence: (XMPPPresence*)presence
387 OFString *tooltip = @"";
389 [[presence elementForName: @"show"
390 namespace: XMPP_NS_CLIENT] stringValue];
392 [[presence elementForName: @"status"
393 namespace: XMPP_NS_CLIENT] stringValue];
396 tooltip = [@"<b>Status:</b> " stringByAppendingString: status];
399 // Block the PresenceComboBox's changed handler, so it doesn't
400 // fire and resend presence
401 g_signal_handler_block(presence_combo,
402 presence_combo_changed_handler_id);
404 if ([presence.type isEqual: @"unavailable"])
405 gtk_combo_box_set_active_id(presence_combo,
407 else if (show == nil)
408 gtk_combo_box_set_active_id(presence_combo,
411 gtk_combo_box_set_active_id(presence_combo,
414 // Unblock the changed handler
415 g_signal_handler_unblock(presence_combo,
416 presence_combo_changed_handler_id);
418 gtk_widget_set_tooltip_markup(GTK_WIDGET(presence_combo),
419 [tooltip UTF8String]);