comparison gtk/gtkroomlist.c @ 14191:009db0b357b5

This is a hand-crafted commit to migrate across subversion revisions 16854:16861, due to some vagaries of the way the original renames were done. Witness that monotone can do in one revision what svn had to spread across several.
author Ethan Blanton <elb@pidgin.im>
date Sat, 16 Dec 2006 04:59:55 +0000
parents
children bee5661c15cb
comparison
equal deleted inserted replaced
14190:366be2ce35a7 14191:009db0b357b5
1 /**
2 * @file gtkroomlist.c GTK+ Room List UI
3 * @ingroup gtkui
4 *
5 * gaim
6 *
7 * Gaim is the legal property of its developers, whose names are too numerous
8 * to list here. Please refer to the COPYRIGHT file distributed with this
9 * source distribution.
10 *
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 */
25
26 #include "internal.h"
27 #include "gtkgaim.h"
28 #include "gtkutils.h"
29 #include "gaimstock.h"
30 #include "debug.h"
31 #include "account.h"
32 #include "connection.h"
33 #include "notify.h"
34
35 #include "gtkroomlist.h"
36
37 typedef struct _GaimGtkRoomlist {
38 GaimGtkRoomlistDialog *dialog;
39 GtkTreeStore *model;
40 GtkWidget *tree;
41 GHashTable *cats; /**< Meow. */
42 gint num_rooms, total_rooms;
43 } GaimGtkRoomlist;
44
45 struct _GaimGtkRoomlistDialog {
46 GtkWidget *window;
47 GtkWidget *account_widget;
48 GtkWidget *progress;
49 GtkWidget *sw;
50
51 GtkWidget *stop_button;
52 GtkWidget *list_button;
53 GtkWidget *add_button;
54 GtkWidget *join_button;
55 GtkWidget *close_button;
56
57 GaimAccount *account;
58 GaimRoomlist *roomlist;
59
60 gboolean pg_needs_pulse;
61 gboolean pg_to_active;
62 guint pg_update_to;
63 };
64
65 enum {
66 NAME_COLUMN = 0,
67 ROOM_COLUMN,
68 NUM_OF_COLUMNS,
69 };
70
71 static GList *roomlists = NULL;
72
73 static gint delete_win_cb(GtkWidget *w, GdkEventAny *e, gpointer d)
74 {
75 GaimGtkRoomlistDialog *dialog;
76
77 dialog = (GaimGtkRoomlistDialog *) d;
78
79 if (dialog->roomlist && gaim_roomlist_get_in_progress(dialog->roomlist))
80 gaim_roomlist_cancel_get_list(dialog->roomlist);
81
82 if (dialog->roomlist) {
83 if (dialog->pg_to_active) {
84 gaim_timeout_remove(dialog->pg_update_to);
85 dialog->pg_to_active = FALSE;
86 /* yes, that's right, unref it twice. */
87 gaim_roomlist_unref(dialog->roomlist);
88 }
89 }
90
91 /* free stuff here */
92 if (dialog->roomlist)
93 gaim_roomlist_unref(dialog->roomlist);
94 g_free(dialog);
95
96 return FALSE;
97 }
98
99 static void dialog_select_account_cb(GObject *w, GaimAccount *account,
100 GaimGtkRoomlistDialog *dialog)
101 {
102 dialog->account = account;
103 }
104
105 static void list_button_cb(GtkButton *button, GaimGtkRoomlistDialog *dialog)
106 {
107 GaimConnection *gc;
108 GaimGtkRoomlist *rl;
109
110 gc = gaim_account_get_connection(dialog->account);
111 if (!gc)
112 return;
113
114 if (dialog->roomlist != NULL) {
115 rl = dialog->roomlist->ui_data;
116 gtk_widget_destroy(rl->tree);
117 gaim_roomlist_unref(dialog->roomlist);
118 }
119
120 dialog->roomlist = gaim_roomlist_get_list(gc);
121 if (!dialog->roomlist)
122 return;
123 gaim_roomlist_ref(dialog->roomlist);
124 rl = dialog->roomlist->ui_data;
125 rl->dialog = dialog;
126
127 if (dialog->account_widget)
128 gtk_widget_set_sensitive(dialog->account_widget, FALSE);
129
130 gtk_container_add(GTK_CONTAINER(dialog->sw), rl->tree);
131
132 gtk_widget_set_sensitive(dialog->stop_button, TRUE);
133 gtk_widget_set_sensitive(dialog->list_button, FALSE);
134 gtk_widget_set_sensitive(dialog->add_button, FALSE);
135 gtk_widget_set_sensitive(dialog->join_button, FALSE);
136 }
137
138 static void stop_button_cb(GtkButton *button, GaimGtkRoomlistDialog *dialog)
139 {
140 gaim_roomlist_cancel_get_list(dialog->roomlist);
141
142 if (dialog->account_widget)
143 gtk_widget_set_sensitive(dialog->account_widget, TRUE);
144
145 gtk_widget_set_sensitive(dialog->stop_button, FALSE);
146 gtk_widget_set_sensitive(dialog->list_button, TRUE);
147 gtk_widget_set_sensitive(dialog->add_button, FALSE);
148 gtk_widget_set_sensitive(dialog->join_button, FALSE);
149 }
150
151 static void close_button_cb(GtkButton *button, GaimGtkRoomlistDialog *dialog)
152 {
153 GtkWidget *window = dialog->window;
154
155 delete_win_cb(NULL, NULL, dialog);
156 gtk_widget_destroy(window);
157 }
158
159 struct _menu_cb_info {
160 GaimRoomlist *list;
161 GaimRoomlistRoom *room;
162 };
163
164 static void
165 selection_changed_cb(GtkTreeSelection *selection, GaimGtkRoomlist *grl) {
166 GtkTreeIter iter;
167 GValue val;
168 GaimRoomlistRoom *room;
169 static struct _menu_cb_info *info;
170 GaimGtkRoomlistDialog *dialog;
171
172 dialog = grl->dialog;
173
174 if (gtk_tree_selection_get_selected(selection, NULL, &iter)) {
175 val.g_type = 0;
176 gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val);
177 room = g_value_get_pointer(&val);
178 if (!room || !(room->type & GAIM_ROOMLIST_ROOMTYPE_ROOM)) {
179 gtk_widget_set_sensitive(dialog->join_button, FALSE);
180 gtk_widget_set_sensitive(dialog->add_button, FALSE);
181 return;
182 }
183
184 info = g_new0(struct _menu_cb_info, 1);
185 info->list = dialog->roomlist;
186 info->room = room;
187
188 g_object_set_data_full(G_OBJECT(dialog->join_button), "room-info",
189 info, g_free);
190 g_object_set_data(G_OBJECT(dialog->add_button), "room-info", info);
191
192 gtk_widget_set_sensitive(dialog->add_button, TRUE);
193 gtk_widget_set_sensitive(dialog->join_button, TRUE);
194 } else {
195 gtk_widget_set_sensitive(dialog->add_button, FALSE);
196 gtk_widget_set_sensitive(dialog->join_button, FALSE);
197 }
198 }
199
200 static void do_add_room_cb(GtkWidget *w, struct _menu_cb_info *info)
201 {
202 gaim_blist_request_add_chat(info->list->account, NULL, NULL, info->room->name);
203 }
204
205 static void add_room_to_blist_cb(GtkButton *button, GaimGtkRoomlistDialog *dialog)
206 {
207 GaimRoomlist *rl = dialog->roomlist;
208 GaimGtkRoomlist *grl = rl->ui_data;
209 struct _menu_cb_info *info;
210
211 info = (struct _menu_cb_info*)g_object_get_data(G_OBJECT(button), "room-info");
212
213 if(info != NULL)
214 do_add_room_cb(grl->tree, info);
215 }
216
217 static void do_join_cb(GtkWidget *w, struct _menu_cb_info *info)
218 {
219 gaim_roomlist_room_join(info->list, info->room);
220 }
221
222 static void join_button_cb(GtkButton *button, GaimGtkRoomlistDialog *dialog)
223 {
224 GaimRoomlist *rl = dialog->roomlist;
225 GaimGtkRoomlist *grl = rl->ui_data;
226 struct _menu_cb_info *info;
227
228 info = (struct _menu_cb_info*)g_object_get_data(G_OBJECT(button), "room-info");
229
230 if(info != NULL)
231 do_join_cb(grl->tree, info);
232 }
233
234 static void row_activated_cb(GtkTreeView *tv, GtkTreePath *path, GtkTreeViewColumn *arg2,
235 GaimRoomlist *list)
236 {
237 GaimGtkRoomlist *grl = list->ui_data;
238 GtkTreeIter iter;
239 GaimRoomlistRoom *room;
240 GValue val;
241 struct _menu_cb_info info;
242
243 gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path);
244 val.g_type = 0;
245 gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val);
246 room = g_value_get_pointer(&val);
247 if (!room || !(room->type & GAIM_ROOMLIST_ROOMTYPE_ROOM))
248 return;
249
250 info.list = list;
251 info.room = room;
252
253 do_join_cb(GTK_WIDGET(tv), &info);
254 }
255
256 static gboolean room_click_cb(GtkWidget *tv, GdkEventButton *event, GaimRoomlist *list)
257 {
258 GtkTreePath *path;
259 GaimGtkRoomlist *grl = list->ui_data;
260 GValue val;
261 GaimRoomlistRoom *room;
262 GtkTreeIter iter;
263 GtkWidget *menu;
264 static struct _menu_cb_info info; /* XXX? */
265
266 if (event->button != 3 || event->type != GDK_BUTTON_PRESS)
267 return FALSE;
268
269 /* Here we figure out which room was clicked */
270 if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), event->x, event->y, &path, NULL, NULL, NULL))
271 return FALSE;
272 gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path);
273 gtk_tree_path_free(path);
274 val.g_type = 0;
275 gtk_tree_model_get_value (GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val);
276 room = g_value_get_pointer(&val);
277
278 if (!room || !(room->type & GAIM_ROOMLIST_ROOMTYPE_ROOM))
279 return FALSE;
280
281 info.list = list;
282 info.room = room;
283
284 menu = gtk_menu_new();
285 gaim_new_item_from_stock(menu, _("_Join"), GAIM_STOCK_CHAT,
286 G_CALLBACK(do_join_cb), &info, 0, 0, NULL);
287 gaim_new_item_from_stock(menu, _("_Add"), GTK_STOCK_ADD,
288 G_CALLBACK(do_add_room_cb), &info, 0, 0, NULL);
289
290 gtk_widget_show_all(menu);
291 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 3, event->time);
292
293 return FALSE;
294 }
295
296 static void row_expanded_cb(GtkTreeView *treeview, GtkTreeIter *arg1, GtkTreePath *arg2, gpointer user_data)
297 {
298 GaimRoomlist *list = user_data;
299 GaimRoomlistRoom *category;
300 GValue val;
301
302 val.g_type = 0;
303 gtk_tree_model_get_value(gtk_tree_view_get_model(treeview), arg1, ROOM_COLUMN, &val);
304 category = g_value_get_pointer(&val);
305
306 if (!category->expanded_once) {
307 gaim_roomlist_expand_category(list, category);
308 category->expanded_once = TRUE;
309 }
310 }
311
312 static gboolean account_filter_func(GaimAccount *account)
313 {
314 GaimConnection *gc = gaim_account_get_connection(account);
315 GaimPluginProtocolInfo *prpl_info = NULL;
316
317 prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
318
319 return (prpl_info->roomlist_get_list != NULL);
320 }
321
322 gboolean
323 gaim_gtk_roomlist_is_showable()
324 {
325 GList *c;
326 GaimConnection *gc;
327
328 for (c = gaim_connections_get_all(); c != NULL; c = c->next) {
329 gc = c->data;
330
331 if (account_filter_func(gaim_connection_get_account(gc)))
332 return TRUE;
333 }
334
335 return FALSE;
336 }
337
338 GaimGtkRoomlistDialog *gaim_gtk_roomlist_dialog_new_with_account(GaimAccount *account)
339 {
340 GaimGtkRoomlistDialog *dialog;
341 GtkWidget *window;
342 GtkWidget *vbox;
343 GtkWidget *vbox2;
344 GtkWidget *account_hbox;
345 GtkWidget *bbox;
346 GtkWidget *label;
347
348 dialog = g_new0(GaimGtkRoomlistDialog, 1);
349 dialog->account = account;
350
351 /* Create the window. */
352 dialog->window = window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
353 gtk_window_set_role(GTK_WINDOW(window), "room list");
354 gtk_window_set_title(GTK_WINDOW(window), _("Room List"));
355
356 gtk_container_set_border_width(GTK_CONTAINER(window), GAIM_HIG_BORDER);
357
358 g_signal_connect(G_OBJECT(window), "delete_event",
359 G_CALLBACK(delete_win_cb), dialog);
360
361 /* Create the parent vbox for everything. */
362 vbox = gtk_vbox_new(FALSE, GAIM_HIG_BORDER);
363 gtk_container_add(GTK_CONTAINER(window), vbox);
364 gtk_widget_show(vbox);
365
366 vbox2 = gtk_vbox_new(FALSE, GAIM_HIG_BORDER);
367 gtk_container_add(GTK_CONTAINER(vbox), vbox2);
368 gtk_widget_show(vbox2);
369
370 /* accounts dropdown list */
371 account_hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
372 gtk_box_pack_start(GTK_BOX(vbox2), account_hbox, FALSE, FALSE, 0);
373 gtk_widget_show(account_hbox);
374
375 label = gtk_label_new(NULL);
376 gtk_box_pack_start(GTK_BOX(account_hbox), label, FALSE, FALSE, 0);
377 gtk_label_set_markup_with_mnemonic(GTK_LABEL(label), _("_Account:"));
378 gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
379 gtk_widget_show(label);
380
381 dialog->account_widget = gaim_gtk_account_option_menu_new(dialog->account, FALSE,
382 G_CALLBACK(dialog_select_account_cb), account_filter_func, dialog);
383
384 if (!dialog->account) /* this is normally null, and we normally don't care what the first selected item is */
385 dialog->account = gaim_gtk_account_option_menu_get_selected(dialog->account_widget);
386
387 gtk_box_pack_start(GTK_BOX(account_hbox), dialog->account_widget, TRUE, TRUE, 0);
388 gtk_label_set_mnemonic_widget(GTK_LABEL(label), GTK_WIDGET(dialog->account_widget));
389 gtk_widget_show(dialog->account_widget);
390
391 /* scrolled window */
392 dialog->sw = gtk_scrolled_window_new(NULL, NULL);
393 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(dialog->sw),
394 GTK_SHADOW_IN);
395 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(dialog->sw),
396 GTK_POLICY_AUTOMATIC,
397 GTK_POLICY_AUTOMATIC);
398 gtk_box_pack_start(GTK_BOX(vbox2), dialog->sw, TRUE, TRUE, 0);
399 gtk_widget_set_size_request(dialog->sw, -1, 250);
400 gtk_widget_show(dialog->sw);
401
402 /* progress bar */
403 dialog->progress = gtk_progress_bar_new();
404 gtk_progress_bar_set_pulse_step(GTK_PROGRESS_BAR(dialog->progress), 0.1);
405 gtk_box_pack_start(GTK_BOX(vbox2), dialog->progress, FALSE, FALSE, 0);
406 gtk_widget_show(dialog->progress);
407
408 /* button box */
409 bbox = gtk_hbutton_box_new();
410 gtk_box_set_spacing(GTK_BOX(bbox), GAIM_HIG_BOX_SPACE);
411 gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
412 gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, TRUE, 0);
413 gtk_widget_show(bbox);
414
415 /* stop button */
416 dialog->stop_button = gtk_button_new_from_stock(GTK_STOCK_STOP);
417 gtk_box_pack_start(GTK_BOX(bbox), dialog->stop_button, FALSE, FALSE, 0);
418 g_signal_connect(G_OBJECT(dialog->stop_button), "clicked",
419 G_CALLBACK(stop_button_cb), dialog);
420 gtk_widget_set_sensitive(dialog->stop_button, FALSE);
421 gtk_widget_show(dialog->stop_button);
422
423 /* list button */
424 dialog->list_button = gaim_pixbuf_button_from_stock(_("_Get List"), GTK_STOCK_REFRESH,
425 GAIM_BUTTON_HORIZONTAL);
426 gtk_box_pack_start(GTK_BOX(bbox), dialog->list_button, FALSE, FALSE, 0);
427 g_signal_connect(G_OBJECT(dialog->list_button), "clicked",
428 G_CALLBACK(list_button_cb), dialog);
429 gtk_widget_show(dialog->list_button);
430
431 /* add button */
432 dialog->add_button = gaim_pixbuf_button_from_stock(_("_Add Chat"), GTK_STOCK_ADD,
433 GAIM_BUTTON_HORIZONTAL);
434 gtk_box_pack_start(GTK_BOX(bbox), dialog->add_button, FALSE, FALSE, 0);
435 g_signal_connect(G_OBJECT(dialog->add_button), "clicked",
436 G_CALLBACK(add_room_to_blist_cb), dialog);
437 gtk_widget_set_sensitive(dialog->add_button, FALSE);
438 gtk_widget_show(dialog->add_button);
439
440 /* join button */
441 dialog->join_button = gaim_pixbuf_button_from_stock(_("_Join"), GAIM_STOCK_CHAT,
442 GAIM_BUTTON_HORIZONTAL);
443 gtk_box_pack_start(GTK_BOX(bbox), dialog->join_button, FALSE, FALSE, 0);
444 g_signal_connect(G_OBJECT(dialog->join_button), "clicked",
445 G_CALLBACK(join_button_cb), dialog);
446 gtk_widget_set_sensitive(dialog->join_button, FALSE);
447 gtk_widget_show(dialog->join_button);
448
449 /* close button */
450 dialog->close_button = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
451 gtk_box_pack_start(GTK_BOX(bbox), dialog->close_button, FALSE, FALSE, 0);
452 g_signal_connect(G_OBJECT(dialog->close_button), "clicked",
453 G_CALLBACK(close_button_cb), dialog);
454 gtk_widget_show(dialog->close_button);
455
456 /* show the dialog window and return the dialog */
457 gtk_widget_show(dialog->window);
458
459 return dialog;
460 }
461
462 GaimGtkRoomlistDialog *gaim_gtk_roomlist_dialog_new(void)
463 {
464 return gaim_gtk_roomlist_dialog_new_with_account(NULL);
465 }
466
467 static void gaim_gtk_roomlist_dialog_show_with_account(GaimAccount *account)
468 {
469 GaimGtkRoomlistDialog *dialog;
470
471 dialog = gaim_gtk_roomlist_dialog_new_with_account(account);
472 if (!dialog)
473 return;
474
475 list_button_cb(GTK_BUTTON(dialog->list_button), dialog);
476 }
477
478 void gaim_gtk_roomlist_dialog_show(void)
479 {
480 gaim_gtk_roomlist_dialog_new();
481 }
482
483 static void gaim_gtk_roomlist_new(GaimRoomlist *list)
484 {
485 GaimGtkRoomlist *rl;
486
487 rl = g_new0(GaimGtkRoomlist, 1);
488
489 list->ui_data = rl;
490
491 rl->cats = g_hash_table_new_full(NULL, NULL, NULL, (GDestroyNotify)gtk_tree_row_reference_free);
492
493 roomlists = g_list_append(roomlists, list);
494 }
495
496 static void int_cell_data_func(GtkTreeViewColumn *col, GtkCellRenderer *renderer,
497 GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data)
498 {
499 gchar buf[16];
500 int myint;
501
502 gtk_tree_model_get(model, iter, GPOINTER_TO_INT(user_data), &myint, -1);
503
504 if (myint)
505 g_snprintf(buf, sizeof(buf), "%d", myint);
506 else
507 buf[0] = '\0';
508
509 g_object_set(renderer, "text", buf, NULL);
510 }
511
512 /* this sorts backwards on purpose, so that clicking name sorts a-z, while clicking users sorts
513 infinity-0. you can still click again to reverse it on any of them. */
514 static gint int_sort_func(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data)
515 {
516 int c, d;
517
518 c = d = 0;
519
520 gtk_tree_model_get(model, a, GPOINTER_TO_INT(user_data), &c, -1);
521 gtk_tree_model_get(model, b, GPOINTER_TO_INT(user_data), &d, -1);
522
523 if (c == d)
524 return 0;
525 else if (c > d)
526 return -1;
527 else
528 return 1;
529 }
530
531 static gboolean
532 _search_func(GtkTreeModel *model, gint column, const gchar *key, GtkTreeIter *iter, gpointer search_data)
533 {
534 gboolean result;
535 gchar *name, *fold, *fkey;
536
537 gtk_tree_model_get(model, iter, column, &name, -1);
538 fold = g_utf8_casefold(name, -1);
539 fkey = g_utf8_casefold(key, -1);
540
541 result = (g_strstr_len(fold, strlen(fold), fkey) == NULL);
542
543 g_free(fold);
544 g_free(fkey);
545 g_free(name);
546
547 return result;
548 }
549
550 static void gaim_gtk_roomlist_set_fields(GaimRoomlist *list, GList *fields)
551 {
552 GaimGtkRoomlist *grl = list->ui_data;
553 gint columns = NUM_OF_COLUMNS;
554 int j;
555 GtkTreeStore *model;
556 GtkWidget *tree;
557 GtkCellRenderer *renderer;
558 GtkTreeViewColumn *column;
559 GtkTreeSelection *selection;
560 GList *l;
561 GType *types;
562
563 g_return_if_fail(grl != NULL);
564
565 columns += g_list_length(fields);
566 types = g_new(GType, columns);
567
568 types[NAME_COLUMN] = G_TYPE_STRING;
569 types[ROOM_COLUMN] = G_TYPE_POINTER;
570
571 for (j = NUM_OF_COLUMNS, l = fields; l; l = l->next, j++) {
572 GaimRoomlistField *f = l->data;
573
574 switch (f->type) {
575 case GAIM_ROOMLIST_FIELD_BOOL:
576 types[j] = G_TYPE_BOOLEAN;
577 break;
578 case GAIM_ROOMLIST_FIELD_INT:
579 types[j] = G_TYPE_INT;
580 break;
581 case GAIM_ROOMLIST_FIELD_STRING:
582 types[j] = G_TYPE_STRING;
583 break;
584 }
585 }
586
587 model = gtk_tree_store_newv(columns, types);
588 g_free(types);
589
590 tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
591 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(tree), TRUE);
592
593 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree));
594 g_signal_connect(G_OBJECT(selection), "changed",
595 G_CALLBACK(selection_changed_cb), grl);
596
597 g_object_unref(model);
598
599 grl->model = model;
600 grl->tree = tree;
601 gtk_widget_show(grl->tree);
602
603 renderer = gtk_cell_renderer_text_new();
604 column = gtk_tree_view_column_new_with_attributes(_("Name"), renderer,
605 "text", NAME_COLUMN, NULL);
606 gtk_tree_view_column_set_sizing(GTK_TREE_VIEW_COLUMN(column),
607 GTK_TREE_VIEW_COLUMN_GROW_ONLY);
608 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE);
609 gtk_tree_view_column_set_sort_column_id(GTK_TREE_VIEW_COLUMN(column), NAME_COLUMN);
610 gtk_tree_view_column_set_reorderable(GTK_TREE_VIEW_COLUMN(column), TRUE);
611 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column);
612
613 for (j = NUM_OF_COLUMNS, l = fields; l; l = l->next, j++) {
614 GaimRoomlistField *f = l->data;
615
616 if (f->hidden)
617 continue;
618
619 renderer = gtk_cell_renderer_text_new();
620 column = gtk_tree_view_column_new_with_attributes(f->label, renderer,
621 "text", j, NULL);
622 gtk_tree_view_column_set_sizing(GTK_TREE_VIEW_COLUMN(column),
623 GTK_TREE_VIEW_COLUMN_GROW_ONLY);
624 gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE);
625 gtk_tree_view_column_set_sort_column_id(GTK_TREE_VIEW_COLUMN(column), j);
626 gtk_tree_view_column_set_reorderable(GTK_TREE_VIEW_COLUMN(column), TRUE);
627 if (f->type == GAIM_ROOMLIST_FIELD_INT) {
628 gtk_tree_view_column_set_cell_data_func(column, renderer, int_cell_data_func,
629 GINT_TO_POINTER(j), NULL);
630 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(model), j, int_sort_func,
631 GINT_TO_POINTER(j), NULL);
632 }
633 gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column);
634 }
635
636 g_signal_connect(G_OBJECT(tree), "button-press-event", G_CALLBACK(room_click_cb), list);
637 g_signal_connect(G_OBJECT(tree), "row-expanded", G_CALLBACK(row_expanded_cb), list);
638 g_signal_connect(G_OBJECT(tree), "row-activated", G_CALLBACK(row_activated_cb), list);
639
640 /* Enable CTRL+F searching */
641 gtk_tree_view_set_search_column(GTK_TREE_VIEW(tree), NAME_COLUMN);
642 gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(tree), _search_func, NULL, NULL);
643
644 }
645
646 static gboolean gaim_gtk_progress_bar_pulse(gpointer data)
647 {
648 GaimRoomlist *list = data;
649 GaimGtkRoomlist *rl = list->ui_data;
650
651 if (!rl || !rl->dialog || !rl->dialog->pg_needs_pulse) {
652 if (rl && rl->dialog)
653 rl->dialog->pg_to_active = FALSE;
654 gaim_roomlist_unref(list);
655 return FALSE;
656 }
657
658 gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress));
659 rl->dialog->pg_needs_pulse = FALSE;
660 return TRUE;
661 }
662
663 static void gaim_gtk_roomlist_add_room(GaimRoomlist *list, GaimRoomlistRoom *room)
664 {
665 GaimGtkRoomlist *rl = list->ui_data;
666 GtkTreeRowReference *rr, *parentrr = NULL;
667 GtkTreePath *path;
668 GtkTreeIter iter, parent, child;
669 GList *l, *k;
670 int j;
671 gboolean append = TRUE;
672
673 rl->total_rooms++;
674 if (room->type == GAIM_ROOMLIST_ROOMTYPE_ROOM)
675 rl->num_rooms++;
676
677 if (rl->dialog) {
678 if (!rl->dialog->pg_to_active) {
679 rl->dialog->pg_to_active = TRUE;
680 gaim_roomlist_ref(list);
681 rl->dialog->pg_update_to = g_timeout_add(100, gaim_gtk_progress_bar_pulse, list);
682 gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress));
683 } else {
684 rl->dialog->pg_needs_pulse = TRUE;
685 }
686 }
687 if (room->parent) {
688 parentrr = g_hash_table_lookup(rl->cats, room->parent);
689 path = gtk_tree_row_reference_get_path(parentrr);
690 if (path) {
691 GaimRoomlistRoom *tmproom = NULL;
692
693 gtk_tree_model_get_iter(GTK_TREE_MODEL(rl->model), &parent, path);
694 gtk_tree_path_free(path);
695
696 if (gtk_tree_model_iter_children(GTK_TREE_MODEL(rl->model), &child, &parent)) {
697 gtk_tree_model_get(GTK_TREE_MODEL(rl->model), &child, ROOM_COLUMN, &tmproom, -1);
698 if (!tmproom)
699 append = FALSE;
700 }
701 }
702 }
703
704 if (append)
705 gtk_tree_store_append(rl->model, &iter, (parentrr ? &parent : NULL));
706 else
707 iter = child;
708
709 if (room->type & GAIM_ROOMLIST_ROOMTYPE_CATEGORY)
710 gtk_tree_store_append(rl->model, &child, &iter);
711
712 path = gtk_tree_model_get_path(GTK_TREE_MODEL(rl->model), &iter);
713
714 if (room->type & GAIM_ROOMLIST_ROOMTYPE_CATEGORY) {
715 rr = gtk_tree_row_reference_new(GTK_TREE_MODEL(rl->model), path);
716 g_hash_table_insert(rl->cats, room, rr);
717 }
718
719 gtk_tree_path_free(path);
720
721 gtk_tree_store_set(rl->model, &iter, NAME_COLUMN, room->name, -1);
722 gtk_tree_store_set(rl->model, &iter, ROOM_COLUMN, room, -1);
723
724 for (j = NUM_OF_COLUMNS, l = room->fields, k = list->fields; l && k; j++, l = l->next, k = k->next) {
725 GaimRoomlistField *f = k->data;
726 if (f->hidden)
727 continue;
728 gtk_tree_store_set(rl->model, &iter, j, l->data, -1);
729 }
730 }
731
732 static void gaim_gtk_roomlist_in_progress(GaimRoomlist *list, gboolean flag)
733 {
734 GaimGtkRoomlist *rl = list->ui_data;
735
736 if (!rl || !rl->dialog)
737 return;
738
739 if (flag) {
740 if (rl->dialog->account_widget)
741 gtk_widget_set_sensitive(rl->dialog->account_widget, FALSE);
742 gtk_widget_set_sensitive(rl->dialog->stop_button, TRUE);
743 gtk_widget_set_sensitive(rl->dialog->list_button, FALSE);
744 } else {
745 rl->dialog->pg_needs_pulse = FALSE;
746 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(rl->dialog->progress), 0.0);
747 if (rl->dialog->account_widget)
748 gtk_widget_set_sensitive(rl->dialog->account_widget, TRUE);
749 gtk_widget_set_sensitive(rl->dialog->stop_button, FALSE);
750 gtk_widget_set_sensitive(rl->dialog->list_button, TRUE);
751 }
752 }
753
754 static void gaim_gtk_roomlist_destroy(GaimRoomlist *list)
755 {
756 GaimGtkRoomlist *rl;
757
758 roomlists = g_list_remove(roomlists, list);
759
760 rl = list->ui_data;
761
762 g_return_if_fail(rl != NULL);
763
764 g_hash_table_destroy(rl->cats);
765 g_free(rl);
766 list->ui_data = NULL;
767 }
768
769 static GaimRoomlistUiOps ops = {
770 gaim_gtk_roomlist_dialog_show_with_account,
771 gaim_gtk_roomlist_new,
772 gaim_gtk_roomlist_set_fields,
773 gaim_gtk_roomlist_add_room,
774 gaim_gtk_roomlist_in_progress,
775 gaim_gtk_roomlist_destroy
776 };
777
778
779 void gaim_gtk_roomlist_init(void)
780 {
781 gaim_roomlist_set_ui_ops(&ops);
782 }