Mercurial > pidgin.yaz
view pidgin/gtkroomlist.c @ 21499:fa407f34864f
merge of 'cf2b9ed82ae35086ec3c7557b3b00c0664daca8e'
and 'eeae2261ffb1dccbc44c10b02a7e95db57406bd9'
author | Luke Schierer <lschiere@pidgin.im> |
---|---|
date | Wed, 14 Nov 2007 12:44:45 +0000 |
parents | a04a0d3f9b4f |
children | 2a2496044eef 29a4b8e5f4f6 |
line wrap: on
line source
/** * @file gtkroomlist.c GTK+ Room List UI * @ingroup pidgin */ /* pidgin * * Pidgin is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * source distribution. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA */ #include "internal.h" #include "pidgin.h" #include "gtkutils.h" #include "pidginstock.h" #include "debug.h" #include "account.h" #include "connection.h" #include "notify.h" #include "gtkroomlist.h" typedef struct _PidginRoomlistDialog { GtkWidget *window; GtkWidget *account_widget; GtkWidget *progress; GtkWidget *sw; GtkWidget *stop_button; GtkWidget *list_button; GtkWidget *add_button; GtkWidget *join_button; GtkWidget *close_button; PurpleAccount *account; PurpleRoomlist *roomlist; gboolean pg_needs_pulse; gboolean pg_to_active; guint pg_update_to; } PidginRoomlistDialog; typedef struct _PidginRoomlist { PidginRoomlistDialog *dialog; GtkTreeStore *model; GtkWidget *tree; GHashTable *cats; /**< Meow. */ gint num_rooms, total_rooms; GtkWidget *tipwindow; GdkRectangle tip_rect; guint timeout; PangoLayout *tip_layout; PangoLayout *tip_name_layout; int tip_height; int tip_width; int tip_name_height; int tip_name_width; } PidginRoomlist; enum { NAME_COLUMN = 0, ROOM_COLUMN, NUM_OF_COLUMNS, }; static GList *roomlists = NULL; static gint delete_win_cb(GtkWidget *w, GdkEventAny *e, gpointer d) { PidginRoomlistDialog *dialog; dialog = (PidginRoomlistDialog *) d; if (dialog->roomlist && purple_roomlist_get_in_progress(dialog->roomlist)) purple_roomlist_cancel_get_list(dialog->roomlist); if (dialog->roomlist) { if (dialog->pg_to_active) { purple_timeout_remove(dialog->pg_update_to); dialog->pg_to_active = FALSE; /* yes, that's right, unref it twice. */ purple_roomlist_unref(dialog->roomlist); } } /* free stuff here */ if (dialog->roomlist) purple_roomlist_unref(dialog->roomlist); g_free(dialog); return FALSE; } static void dialog_select_account_cb(GObject *w, PurpleAccount *account, PidginRoomlistDialog *dialog) { dialog->account = account; } static void list_button_cb(GtkButton *button, PidginRoomlistDialog *dialog) { PurpleConnection *gc; PidginRoomlist *rl; gc = purple_account_get_connection(dialog->account); if (!gc) return; if (dialog->roomlist != NULL) { rl = dialog->roomlist->ui_data; gtk_widget_destroy(rl->tree); purple_roomlist_unref(dialog->roomlist); } dialog->roomlist = purple_roomlist_get_list(gc); if (!dialog->roomlist) return; purple_roomlist_ref(dialog->roomlist); rl = dialog->roomlist->ui_data; rl->dialog = dialog; if (dialog->account_widget) gtk_widget_set_sensitive(dialog->account_widget, FALSE); gtk_container_add(GTK_CONTAINER(dialog->sw), rl->tree); /* some protocols (not bundled with libpurple) finish getting their * room list immediately */ if(purple_roomlist_get_in_progress(dialog->roomlist)) { gtk_widget_set_sensitive(dialog->stop_button, TRUE); gtk_widget_set_sensitive(dialog->list_button, FALSE); } else { gtk_widget_set_sensitive(dialog->stop_button, FALSE); gtk_widget_set_sensitive(dialog->list_button, TRUE); } gtk_widget_set_sensitive(dialog->add_button, FALSE); gtk_widget_set_sensitive(dialog->join_button, FALSE); } static void stop_button_cb(GtkButton *button, PidginRoomlistDialog *dialog) { purple_roomlist_cancel_get_list(dialog->roomlist); if (dialog->account_widget) gtk_widget_set_sensitive(dialog->account_widget, TRUE); gtk_widget_set_sensitive(dialog->stop_button, FALSE); gtk_widget_set_sensitive(dialog->list_button, TRUE); gtk_widget_set_sensitive(dialog->add_button, FALSE); gtk_widget_set_sensitive(dialog->join_button, FALSE); } static void close_button_cb(GtkButton *button, PidginRoomlistDialog *dialog) { GtkWidget *window = dialog->window; delete_win_cb(NULL, NULL, dialog); gtk_widget_destroy(window); } struct _menu_cb_info { PurpleRoomlist *list; PurpleRoomlistRoom *room; }; static void selection_changed_cb(GtkTreeSelection *selection, PidginRoomlist *grl) { GtkTreeIter iter; GValue val; PurpleRoomlistRoom *room; static struct _menu_cb_info *info; PidginRoomlistDialog *dialog; dialog = grl->dialog; if (gtk_tree_selection_get_selected(selection, NULL, &iter)) { val.g_type = 0; gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(room->type & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) { gtk_widget_set_sensitive(dialog->join_button, FALSE); gtk_widget_set_sensitive(dialog->add_button, FALSE); return; } info = g_new0(struct _menu_cb_info, 1); info->list = dialog->roomlist; info->room = room; g_object_set_data_full(G_OBJECT(dialog->join_button), "room-info", info, g_free); g_object_set_data(G_OBJECT(dialog->add_button), "room-info", info); gtk_widget_set_sensitive(dialog->add_button, TRUE); gtk_widget_set_sensitive(dialog->join_button, TRUE); } else { gtk_widget_set_sensitive(dialog->add_button, FALSE); gtk_widget_set_sensitive(dialog->join_button, FALSE); } } static void do_add_room_cb(GtkWidget *w, struct _menu_cb_info *info) { char *name; PurpleConnection *gc = purple_account_get_connection(info->list->account); PurplePluginProtocolInfo *prpl_info = NULL; if(gc != NULL) prpl_info = PURPLE_PLUGIN_PROTOCOL_INFO(gc->prpl); if(prpl_info != NULL && prpl_info->roomlist_room_serialize) name = prpl_info->roomlist_room_serialize(info->room); else name = g_strdup(info->room->name); purple_blist_request_add_chat(info->list->account, NULL, NULL, name); g_free(name); } static void add_room_to_blist_cb(GtkButton *button, PidginRoomlistDialog *dialog) { PurpleRoomlist *rl = dialog->roomlist; PidginRoomlist *grl = rl->ui_data; struct _menu_cb_info *info; info = (struct _menu_cb_info*)g_object_get_data(G_OBJECT(button), "room-info"); if(info != NULL) do_add_room_cb(grl->tree, info); } static void do_join_cb(GtkWidget *w, struct _menu_cb_info *info) { purple_roomlist_room_join(info->list, info->room); } static void join_button_cb(GtkButton *button, PidginRoomlistDialog *dialog) { PurpleRoomlist *rl = dialog->roomlist; PidginRoomlist *grl = rl->ui_data; struct _menu_cb_info *info; info = (struct _menu_cb_info*)g_object_get_data(G_OBJECT(button), "room-info"); if(info != NULL) do_join_cb(grl->tree, info); } static void row_activated_cb(GtkTreeView *tv, GtkTreePath *path, GtkTreeViewColumn *arg2, PurpleRoomlist *list) { PidginRoomlist *grl = list->ui_data; GtkTreeIter iter; PurpleRoomlistRoom *room; GValue val; struct _menu_cb_info info; gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); val.g_type = 0; gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(room->type & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) return; info.list = list; info.room = room; do_join_cb(GTK_WIDGET(tv), &info); } static gboolean room_click_cb(GtkWidget *tv, GdkEventButton *event, PurpleRoomlist *list) { GtkTreePath *path; PidginRoomlist *grl = list->ui_data; GValue val; PurpleRoomlistRoom *room; GtkTreeIter iter; GtkWidget *menu; static struct _menu_cb_info info; /* XXX? */ if (event->button != 3 || event->type != GDK_BUTTON_PRESS) return FALSE; /* Here we figure out which room was clicked */ if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), event->x, event->y, &path, NULL, NULL, NULL)) return FALSE; gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); gtk_tree_path_free(path); val.g_type = 0; gtk_tree_model_get_value (GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(room->type & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) return FALSE; info.list = list; info.room = room; menu = gtk_menu_new(); pidgin_new_item_from_stock(menu, _("_Join"), PIDGIN_STOCK_CHAT, G_CALLBACK(do_join_cb), &info, 0, 0, NULL); pidgin_new_item_from_stock(menu, _("_Add"), GTK_STOCK_ADD, G_CALLBACK(do_add_room_cb), &info, 0, 0, NULL); gtk_widget_show_all(menu); gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 3, event->time); return FALSE; } static void row_expanded_cb(GtkTreeView *treeview, GtkTreeIter *arg1, GtkTreePath *arg2, gpointer user_data) { PurpleRoomlist *list = user_data; PurpleRoomlistRoom *category; GValue val; val.g_type = 0; gtk_tree_model_get_value(gtk_tree_view_get_model(treeview), arg1, ROOM_COLUMN, &val); category = g_value_get_pointer(&val); if (!category->expanded_once) { purple_roomlist_expand_category(list, category); category->expanded_once = TRUE; } } static void pidgin_roomlist_tooltip_destroy(PidginRoomlist *grl) { if ((grl == NULL) || (grl->tipwindow == NULL)) return; gtk_widget_destroy(grl->tipwindow); grl->tipwindow = NULL; } static void pidgin_roomlist_tooltip_destroy_cb(GObject *object, PidginRoomlist *grl) { if ((grl == NULL) || (grl->tipwindow == NULL)) return; if (grl->timeout) g_source_remove(grl->timeout); grl->timeout = 0; pidgin_roomlist_tooltip_destroy(grl); } #define SMALL_SPACE 6 #define TOOLTIP_BORDER 12 static void pidgin_roomlist_paint_tip(GtkWidget *widget, GdkEventExpose *event, gpointer user_data) { PidginRoomlist *grl = (PidginRoomlist *)user_data; GtkStyle *style; int current_height, max_width; int max_text_width; GtkTextDirection dir = gtk_widget_get_direction(GTK_WIDGET(grl->tree)); style = grl->tipwindow->style; gtk_paint_flat_box(style, grl->tipwindow->window, GTK_STATE_NORMAL, GTK_SHADOW_OUT, NULL, grl->tipwindow, "tooltip", 0, 0, -1, -1); max_text_width = 0; max_text_width = MAX(grl->tip_width, grl->tip_name_width); max_width = TOOLTIP_BORDER + SMALL_SPACE + max_text_width + TOOLTIP_BORDER; current_height = 12; if (dir == GTK_TEXT_DIR_RTL) { gtk_paint_layout(style, grl->tipwindow->window, GTK_STATE_NORMAL, FALSE, NULL, grl->tipwindow, "tooltip", max_width - (TOOLTIP_BORDER + SMALL_SPACE) - PANGO_PIXELS(600000), current_height, grl->tip_name_layout); } else { gtk_paint_layout (style, grl->tipwindow->window, GTK_STATE_NORMAL, FALSE, NULL, grl->tipwindow, "tooltip", TOOLTIP_BORDER + SMALL_SPACE, current_height, grl->tip_name_layout); } if (dir != GTK_TEXT_DIR_RTL) { gtk_paint_layout (style, grl->tipwindow->window, GTK_STATE_NORMAL, FALSE, NULL, grl->tipwindow, "tooltip", TOOLTIP_BORDER + SMALL_SPACE, current_height + grl->tip_name_height, grl->tip_layout); } else { gtk_paint_layout(style, grl->tipwindow->window, GTK_STATE_NORMAL, FALSE, NULL, grl->tipwindow, "tooltip", max_width - (TOOLTIP_BORDER + SMALL_SPACE) - PANGO_PIXELS(600000), current_height + grl->tip_name_height, grl->tip_layout); } } static gboolean pidgin_roomlist_create_tip(PurpleRoomlist *list) { PidginRoomlist *grl = list->ui_data; GtkWidget *tv = grl->tree; PurpleRoomlistRoom *room; GtkTreePath *path; GtkTreeIter iter; GValue val; gchar *name, *tmp, *node_name; GString *tooltip_text = NULL; GList *l, *k; gint j; gboolean first = TRUE; if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), grl->tip_rect.x, grl->tip_rect.y + (grl->tip_rect.height/2), &path, NULL, NULL, NULL)) return FALSE; gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); val.g_type = 0; gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(room->type & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) return FALSE; tooltip_text = g_string_new(""); gtk_tree_model_get(GTK_TREE_MODEL(grl->model), &iter, NAME_COLUMN, &name, -1); for (j = NUM_OF_COLUMNS, l = room->fields, k = list->fields; l && k; j++, l = l->next, k = k->next) { PurpleRoomlistField *f = k->data; gchar *label; if (f->hidden) continue; label = g_markup_escape_text(f->label, -1); switch (f->type) { case PURPLE_ROOMLIST_FIELD_BOOL: g_string_append_printf(tooltip_text, "%s<b>%s:</b> %s", first ? "" : "\n", label, l->data ? "True" : "False"); break; case PURPLE_ROOMLIST_FIELD_INT: g_string_append_printf(tooltip_text, "%s<b>%s:</b> %d", first ? "" : "\n", label, GPOINTER_TO_INT(l->data)); break; case PURPLE_ROOMLIST_FIELD_STRING: tmp = g_markup_escape_text((char *)l->data, -1); g_string_append_printf(tooltip_text, "%s<b>%s:</b> %s", first ? "" : "\n", label, tmp); g_free(tmp); break; } first = FALSE; g_free(label); } gtk_tree_path_free(path); grl->tip_layout = gtk_widget_create_pango_layout(grl->tipwindow, NULL); grl->tip_name_layout = gtk_widget_create_pango_layout(grl->tipwindow, NULL); tmp = g_markup_escape_text(name, -1); g_free(name); node_name = g_strdup_printf("<span size='x-large' weight='bold'>%s</span>", tmp); g_free(tmp); pango_layout_set_markup(grl->tip_layout, tooltip_text->str, -1); pango_layout_set_wrap(grl->tip_layout, PANGO_WRAP_WORD); pango_layout_set_width(grl->tip_layout, 600000); pango_layout_get_size (grl->tip_layout, &grl->tip_width, &grl->tip_height); grl->tip_width = PANGO_PIXELS(grl->tip_width); grl->tip_height = PANGO_PIXELS(grl->tip_height); pango_layout_set_markup(grl->tip_name_layout, node_name, -1); pango_layout_set_wrap(grl->tip_name_layout, PANGO_WRAP_WORD); pango_layout_set_width(grl->tip_name_layout, 600000); pango_layout_get_size (grl->tip_name_layout, &grl->tip_name_width, &grl->tip_name_height); grl->tip_name_width = PANGO_PIXELS(grl->tip_name_width) + SMALL_SPACE; grl->tip_name_height = MAX(PANGO_PIXELS(grl->tip_name_height), SMALL_SPACE); g_free(node_name); g_string_free(tooltip_text, TRUE); return TRUE; } static void pidgin_roomlist_draw_tooltip(PurpleRoomlist *list, GtkWidget *widget) { PidginRoomlist *grl = list->ui_data; int scr_w, scr_h, w, h, x, y; #if GTK_CHECK_VERSION(2,2,0) int mon_num; GdkScreen *screen = NULL; #endif GdkRectangle mon_size; int sig; const char *name; pidgin_roomlist_tooltip_destroy(grl); grl->tipwindow = gtk_window_new(GTK_WINDOW_POPUP); gtk_widget_ensure_style (grl->tipwindow); if (!pidgin_roomlist_create_tip(list)) { pidgin_roomlist_tooltip_destroy(grl); return; } name = gtk_window_get_title(GTK_WINDOW(gtk_widget_get_toplevel(widget))); gtk_widget_set_app_paintable(grl->tipwindow, TRUE); gtk_window_set_title(GTK_WINDOW(grl->tipwindow), name ? name : _("Room List")); gtk_window_set_resizable(GTK_WINDOW(grl->tipwindow), FALSE); gtk_widget_set_name(grl->tipwindow, "gtk-tooltips"); g_signal_connect(G_OBJECT(grl->tipwindow), "expose_event", G_CALLBACK(pidgin_roomlist_paint_tip), grl); w = TOOLTIP_BORDER + SMALL_SPACE + MAX(grl->tip_width, grl->tip_name_width) + TOOLTIP_BORDER; h = TOOLTIP_BORDER + grl->tip_height + grl->tip_name_height + TOOLTIP_BORDER; #if GTK_CHECK_VERSION(2,2,0) gdk_display_get_pointer(gdk_display_get_default(), &screen, &x, &y, NULL); mon_num = gdk_screen_get_monitor_at_point(screen, x, y); gdk_screen_get_monitor_geometry(screen, mon_num, &mon_size); scr_w = mon_size.width + mon_size.x; scr_h = mon_size.height + mon_size.y; #else scr_w = gdk_screen_width(); scr_h = gdk_screen_height(); gdk_window_get_pointer(NULL, &x, &y, NULL); mon_size.x = 0; mon_size.y = 0; #endif #if GTK_CHECK_VERSION(2,2,0) if (w > mon_size.width) w = mon_size.width - 10; if (h > mon_size.height) h = mon_size.height - 10; #endif x -= ((w >> 1) + 4); if ((y + h + 4) > scr_h) y = y - h - 5; else y = y + 6; if (y < mon_size.y) y = mon_size.y; if (y != mon_size.y) { if ((x + w) > scr_w) x -= (x + w + 5) - scr_w; else if (x < mon_size.x) x = mon_size.x; } else { x -= (w / 2 + 10); if (x < mon_size.x) x = mon_size.x; } gtk_widget_set_size_request(grl->tipwindow, w, h); gtk_window_move(GTK_WINDOW(grl->tipwindow), x, y); gtk_widget_show(grl->tipwindow); /* Hide the tooltip when the widget is destroyed */ sig = g_signal_connect(G_OBJECT(widget), "destroy", G_CALLBACK(pidgin_roomlist_tooltip_destroy_cb), grl); g_signal_connect_swapped(G_OBJECT(grl->tipwindow), "destroy", G_CALLBACK(g_source_remove), GINT_TO_POINTER(sig)); } static gboolean pidgin_roomlist_tooltip_timeout(PurpleRoomlist *list) { PidginRoomlist *grl = list->ui_data; GtkWidget *tv = grl->tree; GtkTreePath *path; pidgin_roomlist_tooltip_destroy(grl); if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), grl->tip_rect.x, grl->tip_rect.y + (grl->tip_rect.height/2), &path, NULL, NULL, NULL)) return FALSE; pidgin_roomlist_draw_tooltip(list, GTK_WIDGET(grl->tree)); return FALSE; } static gboolean row_motion_cb(GtkWidget *tv, GdkEventMotion *event, gpointer user_data) { PurpleRoomlist *list = user_data; PidginRoomlist *grl = list->ui_data; GtkTreePath *path; int delay; /* XXX: should this be using the blist delay pref? */ delay = purple_prefs_get_int(PIDGIN_PREFS_ROOT "/blist/tooltip_delay"); if (delay == 0) return FALSE; if (grl->timeout) { if ((event->y > grl->tip_rect.y) && ((event->y - grl->tip_rect.height) < grl->tip_rect.y)) return FALSE; /* We've left the cell. Remove the timeout and create a new one below */ pidgin_roomlist_tooltip_destroy(grl); } gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), event->x, event->y, &path, NULL, NULL, NULL); if (path == NULL) { pidgin_roomlist_tooltip_destroy(grl); return FALSE; } gtk_tree_view_get_cell_area(GTK_TREE_VIEW(tv), path, NULL, &grl->tip_rect); if (path) gtk_tree_path_free(path); grl->timeout = g_timeout_add(delay, (GSourceFunc)pidgin_roomlist_tooltip_timeout, list); return FALSE; } static void row_leave_cb(GtkWidget *tv, GdkEventCrossing *e, gpointer user_data) { PurpleRoomlist *list = user_data; PidginRoomlist *grl = list->ui_data; if (grl->timeout) { g_source_remove(grl->timeout); grl->timeout = 0; } pidgin_roomlist_tooltip_destroy(grl); } static gboolean account_filter_func(PurpleAccount *account) { PurpleConnection *gc = purple_account_get_connection(account); PurplePluginProtocolInfo *prpl_info = NULL; prpl_info = PURPLE_PLUGIN_PROTOCOL_INFO(gc->prpl); return (prpl_info->roomlist_get_list != NULL); } gboolean pidgin_roomlist_is_showable() { GList *c; PurpleConnection *gc; for (c = purple_connections_get_all(); c != NULL; c = c->next) { gc = c->data; if (account_filter_func(purple_connection_get_account(gc))) return TRUE; } return FALSE; } static PidginRoomlistDialog * pidgin_roomlist_dialog_new_with_account(PurpleAccount *account) { PidginRoomlistDialog *dialog; GtkWidget *window; GtkWidget *vbox; GtkWidget *vbox2; GtkWidget *account_hbox; GtkWidget *bbox; GtkWidget *label; dialog = g_new0(PidginRoomlistDialog, 1); dialog->account = account; /* Create the window. */ dialog->window = window = pidgin_create_window(_("Room List"), PIDGIN_HIG_BORDER, "room list", TRUE); g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(delete_win_cb), dialog); /* Create the parent vbox for everything. */ vbox = gtk_vbox_new(FALSE, PIDGIN_HIG_BORDER); gtk_container_add(GTK_CONTAINER(window), vbox); gtk_widget_show(vbox); vbox2 = gtk_vbox_new(FALSE, PIDGIN_HIG_BORDER); gtk_container_add(GTK_CONTAINER(vbox), vbox2); gtk_widget_show(vbox2); /* accounts dropdown list */ account_hbox = gtk_hbox_new(FALSE, PIDGIN_HIG_BOX_SPACE); gtk_box_pack_start(GTK_BOX(vbox2), account_hbox, FALSE, FALSE, 0); gtk_widget_show(account_hbox); label = gtk_label_new(NULL); gtk_box_pack_start(GTK_BOX(account_hbox), label, FALSE, FALSE, 0); gtk_label_set_markup_with_mnemonic(GTK_LABEL(label), _("_Account:")); gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); gtk_widget_show(label); dialog->account_widget = pidgin_account_option_menu_new(dialog->account, FALSE, G_CALLBACK(dialog_select_account_cb), account_filter_func, dialog); if (!dialog->account) /* this is normally null, and we normally don't care what the first selected item is */ dialog->account = pidgin_account_option_menu_get_selected(dialog->account_widget); gtk_box_pack_start(GTK_BOX(account_hbox), dialog->account_widget, TRUE, TRUE, 0); gtk_label_set_mnemonic_widget(GTK_LABEL(label), GTK_WIDGET(dialog->account_widget)); gtk_widget_show(dialog->account_widget); /* scrolled window */ dialog->sw = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(dialog->sw), GTK_SHADOW_IN); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(dialog->sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); gtk_box_pack_start(GTK_BOX(vbox2), dialog->sw, TRUE, TRUE, 0); gtk_widget_set_size_request(dialog->sw, -1, 250); gtk_widget_show(dialog->sw); /* progress bar */ dialog->progress = gtk_progress_bar_new(); gtk_progress_bar_set_pulse_step(GTK_PROGRESS_BAR(dialog->progress), 0.1); gtk_box_pack_start(GTK_BOX(vbox2), dialog->progress, FALSE, FALSE, 0); gtk_widget_show(dialog->progress); /* button box */ bbox = gtk_hbutton_box_new(); gtk_box_set_spacing(GTK_BOX(bbox), PIDGIN_HIG_BOX_SPACE); gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END); gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, TRUE, 0); gtk_widget_show(bbox); /* stop button */ dialog->stop_button = gtk_button_new_from_stock(GTK_STOCK_STOP); gtk_box_pack_start(GTK_BOX(bbox), dialog->stop_button, FALSE, FALSE, 0); g_signal_connect(G_OBJECT(dialog->stop_button), "clicked", G_CALLBACK(stop_button_cb), dialog); gtk_widget_set_sensitive(dialog->stop_button, FALSE); gtk_widget_show(dialog->stop_button); /* list button */ dialog->list_button = pidgin_pixbuf_button_from_stock(_("_Get List"), GTK_STOCK_REFRESH, PIDGIN_BUTTON_HORIZONTAL); gtk_box_pack_start(GTK_BOX(bbox), dialog->list_button, FALSE, FALSE, 0); g_signal_connect(G_OBJECT(dialog->list_button), "clicked", G_CALLBACK(list_button_cb), dialog); gtk_widget_show(dialog->list_button); /* add button */ dialog->add_button = pidgin_pixbuf_button_from_stock(_("_Add Chat"), GTK_STOCK_ADD, PIDGIN_BUTTON_HORIZONTAL); gtk_box_pack_start(GTK_BOX(bbox), dialog->add_button, FALSE, FALSE, 0); g_signal_connect(G_OBJECT(dialog->add_button), "clicked", G_CALLBACK(add_room_to_blist_cb), dialog); gtk_widget_set_sensitive(dialog->add_button, FALSE); gtk_widget_show(dialog->add_button); /* join button */ dialog->join_button = pidgin_pixbuf_button_from_stock(_("_Join"), PIDGIN_STOCK_CHAT, PIDGIN_BUTTON_HORIZONTAL); gtk_box_pack_start(GTK_BOX(bbox), dialog->join_button, FALSE, FALSE, 0); g_signal_connect(G_OBJECT(dialog->join_button), "clicked", G_CALLBACK(join_button_cb), dialog); gtk_widget_set_sensitive(dialog->join_button, FALSE); gtk_widget_show(dialog->join_button); /* close button */ dialog->close_button = gtk_button_new_from_stock(GTK_STOCK_CLOSE); gtk_box_pack_start(GTK_BOX(bbox), dialog->close_button, FALSE, FALSE, 0); g_signal_connect(G_OBJECT(dialog->close_button), "clicked", G_CALLBACK(close_button_cb), dialog); gtk_widget_show(dialog->close_button); /* show the dialog window and return the dialog */ gtk_widget_show(dialog->window); return dialog; } void pidgin_roomlist_dialog_show_with_account(PurpleAccount *account) { PidginRoomlistDialog *dialog; dialog = pidgin_roomlist_dialog_new_with_account(account); if (!dialog) return; list_button_cb(GTK_BUTTON(dialog->list_button), dialog); } void pidgin_roomlist_dialog_show(void) { pidgin_roomlist_dialog_new_with_account(NULL); } static void pidgin_roomlist_new(PurpleRoomlist *list) { PidginRoomlist *rl; rl = g_new0(PidginRoomlist, 1); list->ui_data = rl; rl->cats = g_hash_table_new_full(NULL, NULL, NULL, (GDestroyNotify)gtk_tree_row_reference_free); roomlists = g_list_append(roomlists, list); } static void int_cell_data_func(GtkTreeViewColumn *col, GtkCellRenderer *renderer, GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data) { gchar buf[16]; int myint; gtk_tree_model_get(model, iter, GPOINTER_TO_INT(user_data), &myint, -1); if (myint) g_snprintf(buf, sizeof(buf), "%d", myint); else buf[0] = '\0'; g_object_set(renderer, "text", buf, NULL); } /* this sorts backwards on purpose, so that clicking name sorts a-z, while clicking users sorts infinity-0. you can still click again to reverse it on any of them. */ static gint int_sort_func(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data) { int c, d; c = d = 0; gtk_tree_model_get(model, a, GPOINTER_TO_INT(user_data), &c, -1); gtk_tree_model_get(model, b, GPOINTER_TO_INT(user_data), &d, -1); if (c == d) return 0; else if (c > d) return -1; else return 1; } static gboolean _search_func(GtkTreeModel *model, gint column, const gchar *key, GtkTreeIter *iter, gpointer search_data) { gboolean result; gchar *name, *fold, *fkey; gtk_tree_model_get(model, iter, column, &name, -1); fold = g_utf8_casefold(name, -1); fkey = g_utf8_casefold(key, -1); result = (g_strstr_len(fold, strlen(fold), fkey) == NULL); g_free(fold); g_free(fkey); g_free(name); return result; } static void pidgin_roomlist_set_fields(PurpleRoomlist *list, GList *fields) { PidginRoomlist *grl = list->ui_data; gint columns = NUM_OF_COLUMNS; int j; GtkTreeStore *model; GtkWidget *tree; GtkCellRenderer *renderer; GtkTreeViewColumn *column; GtkTreeSelection *selection; GList *l; GType *types; g_return_if_fail(grl != NULL); columns += g_list_length(fields); types = g_new(GType, columns); types[NAME_COLUMN] = G_TYPE_STRING; types[ROOM_COLUMN] = G_TYPE_POINTER; for (j = NUM_OF_COLUMNS, l = fields; l; l = l->next, j++) { PurpleRoomlistField *f = l->data; switch (f->type) { case PURPLE_ROOMLIST_FIELD_BOOL: types[j] = G_TYPE_BOOLEAN; break; case PURPLE_ROOMLIST_FIELD_INT: types[j] = G_TYPE_INT; break; case PURPLE_ROOMLIST_FIELD_STRING: types[j] = G_TYPE_STRING; break; } } model = gtk_tree_store_newv(columns, types); g_free(types); tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model)); gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(tree), TRUE); selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree)); g_signal_connect(G_OBJECT(selection), "changed", G_CALLBACK(selection_changed_cb), grl); g_object_unref(model); grl->model = model; grl->tree = tree; gtk_widget_show(grl->tree); renderer = gtk_cell_renderer_text_new(); column = gtk_tree_view_column_new_with_attributes(_("Name"), renderer, "text", NAME_COLUMN, NULL); gtk_tree_view_column_set_sizing(GTK_TREE_VIEW_COLUMN(column), GTK_TREE_VIEW_COLUMN_GROW_ONLY); gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); gtk_tree_view_column_set_sort_column_id(GTK_TREE_VIEW_COLUMN(column), NAME_COLUMN); gtk_tree_view_column_set_reorderable(GTK_TREE_VIEW_COLUMN(column), TRUE); gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); for (j = NUM_OF_COLUMNS, l = fields; l; l = l->next, j++) { PurpleRoomlistField *f = l->data; if (f->hidden) continue; renderer = gtk_cell_renderer_text_new(); column = gtk_tree_view_column_new_with_attributes(f->label, renderer, "text", j, NULL); gtk_tree_view_column_set_sizing(GTK_TREE_VIEW_COLUMN(column), GTK_TREE_VIEW_COLUMN_GROW_ONLY); gtk_tree_view_column_set_resizable(GTK_TREE_VIEW_COLUMN(column), TRUE); gtk_tree_view_column_set_sort_column_id(GTK_TREE_VIEW_COLUMN(column), j); gtk_tree_view_column_set_reorderable(GTK_TREE_VIEW_COLUMN(column), TRUE); if (f->type == PURPLE_ROOMLIST_FIELD_INT) { gtk_tree_view_column_set_cell_data_func(column, renderer, int_cell_data_func, GINT_TO_POINTER(j), NULL); gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(model), j, int_sort_func, GINT_TO_POINTER(j), NULL); } gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); } g_signal_connect(G_OBJECT(tree), "button-press-event", G_CALLBACK(room_click_cb), list); g_signal_connect(G_OBJECT(tree), "row-expanded", G_CALLBACK(row_expanded_cb), list); g_signal_connect(G_OBJECT(tree), "row-activated", G_CALLBACK(row_activated_cb), list); g_signal_connect(G_OBJECT(tree), "motion-notify-event", G_CALLBACK(row_motion_cb), list); g_signal_connect(G_OBJECT(tree), "leave-notify-event", G_CALLBACK(row_leave_cb), list); /* Enable CTRL+F searching */ gtk_tree_view_set_search_column(GTK_TREE_VIEW(tree), NAME_COLUMN); gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(tree), _search_func, NULL, NULL); } static gboolean pidgin_progress_bar_pulse(gpointer data) { PurpleRoomlist *list = data; PidginRoomlist *rl = list->ui_data; if (!rl || !rl->dialog || !rl->dialog->pg_needs_pulse) { if (rl && rl->dialog) rl->dialog->pg_to_active = FALSE; purple_roomlist_unref(list); return FALSE; } gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress)); rl->dialog->pg_needs_pulse = FALSE; return TRUE; } static void pidgin_roomlist_add_room(PurpleRoomlist *list, PurpleRoomlistRoom *room) { PidginRoomlist *rl = list->ui_data; GtkTreeRowReference *rr, *parentrr = NULL; GtkTreePath *path; GtkTreeIter iter, parent, child; GList *l, *k; int j; gboolean append = TRUE; rl->total_rooms++; if (room->type == PURPLE_ROOMLIST_ROOMTYPE_ROOM) rl->num_rooms++; if (rl->dialog) { if (!rl->dialog->pg_to_active) { rl->dialog->pg_to_active = TRUE; purple_roomlist_ref(list); rl->dialog->pg_update_to = g_timeout_add(100, pidgin_progress_bar_pulse, list); gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress)); } else { rl->dialog->pg_needs_pulse = TRUE; } } if (room->parent) { parentrr = g_hash_table_lookup(rl->cats, room->parent); path = gtk_tree_row_reference_get_path(parentrr); if (path) { PurpleRoomlistRoom *tmproom = NULL; gtk_tree_model_get_iter(GTK_TREE_MODEL(rl->model), &parent, path); gtk_tree_path_free(path); if (gtk_tree_model_iter_children(GTK_TREE_MODEL(rl->model), &child, &parent)) { gtk_tree_model_get(GTK_TREE_MODEL(rl->model), &child, ROOM_COLUMN, &tmproom, -1); if (!tmproom) append = FALSE; } } } if (append) gtk_tree_store_append(rl->model, &iter, (parentrr ? &parent : NULL)); else iter = child; if (room->type & PURPLE_ROOMLIST_ROOMTYPE_CATEGORY) gtk_tree_store_append(rl->model, &child, &iter); path = gtk_tree_model_get_path(GTK_TREE_MODEL(rl->model), &iter); if (room->type & PURPLE_ROOMLIST_ROOMTYPE_CATEGORY) { rr = gtk_tree_row_reference_new(GTK_TREE_MODEL(rl->model), path); g_hash_table_insert(rl->cats, room, rr); } gtk_tree_path_free(path); gtk_tree_store_set(rl->model, &iter, NAME_COLUMN, room->name, -1); gtk_tree_store_set(rl->model, &iter, ROOM_COLUMN, room, -1); for (j = NUM_OF_COLUMNS, l = room->fields, k = list->fields; l && k; j++, l = l->next, k = k->next) { PurpleRoomlistField *f = k->data; if (f->hidden) continue; gtk_tree_store_set(rl->model, &iter, j, l->data, -1); } } static void pidgin_roomlist_in_progress(PurpleRoomlist *list, gboolean flag) { PidginRoomlist *rl = list->ui_data; if (!rl || !rl->dialog) return; if (flag) { if (rl->dialog->account_widget) gtk_widget_set_sensitive(rl->dialog->account_widget, FALSE); gtk_widget_set_sensitive(rl->dialog->stop_button, TRUE); gtk_widget_set_sensitive(rl->dialog->list_button, FALSE); } else { rl->dialog->pg_needs_pulse = FALSE; gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(rl->dialog->progress), 0.0); if (rl->dialog->account_widget) gtk_widget_set_sensitive(rl->dialog->account_widget, TRUE); gtk_widget_set_sensitive(rl->dialog->stop_button, FALSE); gtk_widget_set_sensitive(rl->dialog->list_button, TRUE); } } static void pidgin_roomlist_destroy(PurpleRoomlist *list) { PidginRoomlist *rl; roomlists = g_list_remove(roomlists, list); rl = list->ui_data; g_return_if_fail(rl != NULL); g_hash_table_destroy(rl->cats); g_free(rl); list->ui_data = NULL; } static PurpleRoomlistUiOps ops = { pidgin_roomlist_dialog_show_with_account, pidgin_roomlist_new, pidgin_roomlist_set_fields, pidgin_roomlist_add_room, pidgin_roomlist_in_progress, pidgin_roomlist_destroy, NULL, NULL, NULL, NULL }; void pidgin_roomlist_init(void) { purple_roomlist_set_ui_ops(&ops); }