* 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 * 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 "pidginaccountchooser.h" #include "pidgintooltip.h" GtkWidget *account_widget; PurpleRoomlist *roomlist; PidginRoomlistDialog *dialog; GHashTable *cats; /* Meow. */ gint num_rooms, total_rooms; PangoLayout *tip_name_layout; static GList *roomlists = NULL; static gint delete_win_cb(GtkWidget *w, GdkEventAny *e, gpointer d) PidginRoomlistDialog *dialog = d; if (dialog->roomlist && purple_roomlist_get_in_progress(dialog->roomlist)) purple_roomlist_cancel_get_list(dialog->roomlist); if (dialog->pg_update_to > 0) g_source_remove(dialog->pg_update_to); PidginRoomlist *rl = purple_roomlist_get_ui_data(dialog->roomlist); if (dialog->pg_update_to > 0) /* yes, that's right, unref it twice. */ g_object_unref(dialog->roomlist); g_object_unref(dialog->roomlist); dialog_select_account_cb(GtkWidget *chooser, PidginRoomlistDialog *dialog) PurpleAccount *account = pidgin_account_chooser_get_selected(chooser); gboolean change = (account != dialog->account); dialog->account = account; if (change && dialog->roomlist) { PidginRoomlist *rl = purple_roomlist_get_ui_data(dialog->roomlist); gtk_widget_destroy(rl->tree); g_object_unref(dialog->roomlist); static void list_button_cb(GtkButton *button, PidginRoomlistDialog *dialog) gc = purple_account_get_connection(dialog->account); if (dialog->roomlist != NULL) { rl = purple_roomlist_get_ui_data(dialog->roomlist); gtk_widget_destroy(rl->tree); g_object_unref(dialog->roomlist); dialog->roomlist = purple_roomlist_get_list(gc); g_object_ref(dialog->roomlist); rl = purple_roomlist_get_ui_data(dialog->roomlist); 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); 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); PurpleRoomlistRoom *room; selection_changed_cb(GtkTreeSelection *selection, PidginRoomlist *grl) { PurpleRoomlistRoom *room; static struct _menu_cb_info *info; PidginRoomlistDialog *dialog = grl->dialog; if (gtk_tree_selection_get_selected(selection, NULL, &iter)) { gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(purple_roomlist_room_get_room_type(room) & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) { gtk_widget_set_sensitive(dialog->join_button, FALSE); gtk_widget_set_sensitive(dialog->add_button, FALSE); info = g_new0(struct _menu_cb_info, 1); info->list = dialog->roomlist; g_object_set_data_full(G_OBJECT(dialog->join_button), "room-info", 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); 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) PurpleAccount *account = purple_roomlist_get_account(info->list); PurpleConnection *gc = purple_account_get_connection(account); PurpleProtocol *protocol = NULL; protocol = purple_connection_get_protocol(gc); if(protocol != NULL && PURPLE_PROTOCOL_IMPLEMENTS(protocol, ROOMLIST, room_serialize)) name = purple_protocol_roomlist_iface_room_serialize(protocol, info->room); name = g_strdup(purple_roomlist_room_get_name(info->room)); purple_blist_request_add_chat(account, NULL, NULL, name); static void add_room_to_blist_cb(GtkButton *button, PidginRoomlistDialog *dialog) PurpleRoomlist *rl = dialog->roomlist; PidginRoomlist *grl = purple_roomlist_get_ui_data(rl); struct _menu_cb_info *info = g_object_get_data(G_OBJECT(button), "room-info"); 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 = purple_roomlist_get_ui_data(rl); struct _menu_cb_info *info = g_object_get_data(G_OBJECT(button), "room-info"); do_join_cb(grl->tree, info); static void row_activated_cb(GtkTreeView *tv, GtkTreePath *path, GtkTreeViewColumn *arg2, PidginRoomlist *grl = purple_roomlist_get_ui_data(list); PurpleRoomlistRoom *room; struct _menu_cb_info info; gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(purple_roomlist_room_get_room_type(room) & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) do_join_cb(GTK_WIDGET(tv), &info); static gboolean room_click_cb(GtkWidget *tv, GdkEventButton *event, PurpleRoomlist *list) PidginRoomlist *grl = purple_roomlist_get_ui_data(list); PurpleRoomlistRoom *room; static struct _menu_cb_info info; /* XXX? */ if (!gdk_event_triggers_context_menu((GdkEvent *)event)) /* 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)) gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); gtk_tree_path_free(path); gtk_tree_model_get_value (GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(purple_roomlist_room_get_room_type(room) & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) pidgin_new_menu_item(menu, _("_Join"), PIDGIN_STOCK_CHAT, G_CALLBACK(do_join_cb), &info); pidgin_new_menu_item(menu, _("_Add"), GTK_STOCK_ADD, G_CALLBACK(do_add_room_cb), &info); gtk_widget_show_all(menu); gtk_menu_popup_at_pointer(GTK_MENU(menu), (GdkEvent *)event); static void row_expanded_cb(GtkTreeView *treeview, GtkTreeIter *arg1, GtkTreePath *arg2, gpointer user_data) PurpleRoomlist *list = user_data; PurpleRoomlistRoom *category; gtk_tree_model_get_value(gtk_tree_view_get_model(treeview), arg1, ROOM_COLUMN, &val); category = g_value_get_pointer(&val); if (!purple_roomlist_room_get_expanded_once(category)) { purple_roomlist_expand_category(list, category); purple_roomlist_room_set_expanded_once(category, TRUE); #define TOOLTIP_BORDER 12 pidgin_roomlist_paint_tooltip(GtkWidget *widget, cairo_t *cr, gpointer user_data) PurpleRoomlist *list = user_data; PidginRoomlist *grl = purple_roomlist_get_ui_data(list); int current_height, max_width; GtkTextDirection dir = gtk_widget_get_direction(GTK_WIDGET(grl->tree)); GtkStyleContext *context; context = gtk_widget_get_style_context(grl->tipwindow); gtk_style_context_add_class(context, GTK_STYLE_CLASS_TOOLTIP); max_text_width = MAX(grl->tip_width, grl->tip_name_width); max_width = TOOLTIP_BORDER + SMALL_SPACE + max_text_width + TOOLTIP_BORDER; if (dir == GTK_TEXT_DIR_RTL) { gtk_render_layout(context, cr, max_width - (TOOLTIP_BORDER + SMALL_SPACE) - PANGO_PIXELS(600000), gtk_render_layout(context, cr, TOOLTIP_BORDER + SMALL_SPACE, if (dir != GTK_TEXT_DIR_RTL) { gtk_render_layout(context, cr, TOOLTIP_BORDER + SMALL_SPACE, current_height + grl->tip_name_height, gtk_render_layout(context, cr, max_width - (TOOLTIP_BORDER + SMALL_SPACE) - PANGO_PIXELS(600000), current_height + grl->tip_name_height, static gboolean pidgin_roomlist_create_tip(PurpleRoomlist *list, GtkTreePath *path) PidginRoomlist *grl = purple_roomlist_get_ui_data(list); PurpleRoomlistRoom *room; gchar *name, *tmp, *node_name; GString *tooltip_text = NULL; 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)) gtk_tree_model_get_iter(GTK_TREE_MODEL(grl->model), &iter, path); gtk_tree_model_get_value(GTK_TREE_MODEL(grl->model), &iter, ROOM_COLUMN, &val); room = g_value_get_pointer(&val); if (!room || !(purple_roomlist_room_get_room_type(room) & PURPLE_ROOMLIST_ROOMTYPE_ROOM)) tooltip_text = g_string_new(""); gtk_tree_model_get(GTK_TREE_MODEL(grl->model), &iter, NAME_COLUMN, &name, -1); l = purple_roomlist_room_get_fields(room), k = purple_roomlist_get_fields(list); l && k; j++, l = l->next, k = k->next) PurpleRoomlistField *f = k->data; if (purple_roomlist_field_get_hidden(f)) label = g_markup_escape_text(purple_roomlist_field_get_label(f), -1); switch (purple_roomlist_field_get_field_type(f)) { case PURPLE_ROOMLIST_FIELD_BOOL: g_string_append_printf(tooltip_text, "%s<b>%s:</b> %s", first ? "" : "\n", label, l->data ? "True" : "False"); case PURPLE_ROOMLIST_FIELD_INT: g_string_append_printf(tooltip_text, "%s<b>%s:</b> %d", first ? "" : "\n", label, GPOINTER_TO_INT(l->data)); 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); 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); node_name = g_strdup_printf("<span size='x-large' weight='bold'>%s</span>", 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_string_free(tooltip_text, TRUE); pidgin_roomlist_create_tooltip(GtkWidget *widget, GtkTreePath *path, gpointer data, int *w, int *h) PurpleRoomlist *list = data; PidginRoomlist *grl = purple_roomlist_get_ui_data(list); if (!pidgin_roomlist_create_tip(data, path)) *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 static gboolean account_filter_func(PurpleAccount *account) PurpleConnection *conn = purple_account_get_connection(account); PurpleProtocol *protocol = NULL; if (conn && PURPLE_CONNECTION_IS_CONNECTED(conn)) protocol = purple_connection_get_protocol(conn); return (protocol && PURPLE_PROTOCOL_IMPLEMENTS(protocol, ROOMLIST, get_list)); pidgin_roomlist_is_showable() for (c = purple_connections_get_all(); c != NULL; c = c->next) { if (account_filter_func(purple_connection_get_account(gc))) static PidginRoomlistDialog * pidgin_roomlist_dialog_new_with_account(PurpleAccount *account) PidginRoomlistDialog *dialog; GtkWidget *window, *vbox, *vbox2, *bbox; dialog = g_new0(PidginRoomlistDialog, 1); dialog->account = account; dialog->window = window = pidgin_create_dialog(_("Room List"), 0, "room list", TRUE); g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(delete_win_cb), dialog); /* Create the parent vbox for everything. */ vbox = pidgin_dialog_get_vbox_with_properties(GTK_DIALOG(window), FALSE, PIDGIN_HIG_BORDER); vbox2 = gtk_box_new(GTK_ORIENTATION_VERTICAL, PIDGIN_HIG_BORDER); gtk_container_add(GTK_CONTAINER(vbox), vbox2); /* accounts dropdown list */ pidgin_account_chooser_new(dialog->account, FALSE); pidgin_account_chooser_set_filter_func( PIDGIN_ACCOUNT_CHOOSER(dialog->account_widget), g_signal_connect(dialog->account_widget, "changed", G_CALLBACK(dialog_select_account_cb), dialog); if (!dialog->account) /* this is normally null, and we normally don't care what the first selected item is */ dialog->account = pidgin_account_chooser_get_selected( pidgin_add_widget_to_vbox(GTK_BOX(vbox2), _("_Account:"), NULL, dialog->account_widget, TRUE, NULL); dialog->sw = pidgin_make_scrollable(NULL, GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC, GTK_SHADOW_IN, -1, 250); gtk_box_pack_start(GTK_BOX(vbox2), dialog->sw, TRUE, TRUE, 0); 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); bbox = pidgin_dialog_get_action_area(GTK_DIALOG(window)); gtk_box_set_spacing(GTK_BOX(bbox), PIDGIN_HIG_BOX_SPACE); gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END); dialog->stop_button = pidgin_dialog_add_button(GTK_DIALOG(window), GTK_STOCK_STOP, G_CALLBACK(stop_button_cb), dialog); gtk_widget_set_sensitive(dialog->stop_button, FALSE); 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); 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); 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); dialog->close_button = pidgin_dialog_add_button(GTK_DIALOG(window), GTK_STOCK_CLOSE, G_CALLBACK(close_button_cb), dialog); /* show the dialog window and return the dialog */ gtk_widget_show(dialog->window); void pidgin_roomlist_dialog_show_with_account(PurpleAccount *account) PidginRoomlistDialog *dialog = pidgin_roomlist_dialog_new_with_account(account); 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 = g_new0(PidginRoomlist, 1); purple_roomlist_set_ui_data(list, 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) gtk_tree_model_get(model, iter, GPOINTER_TO_INT(user_data), &myint, -1); g_snprintf(buf, sizeof(buf), "%d", myint); 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) 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); _search_func(GtkTreeModel *model, gint column, const gchar *key, GtkTreeIter *iter, gpointer search_data) 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); static void pidgin_roomlist_set_fields(PurpleRoomlist *list, GList *fields) PidginRoomlist *grl = purple_roomlist_get_ui_data(list); gint columns = NUM_OF_COLUMNS; GtkCellRenderer *renderer; GtkTreeViewColumn *column; GtkTreeSelection *selection; 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 (purple_roomlist_field_get_field_type(f)) { case PURPLE_ROOMLIST_FIELD_BOOL: types[j] = G_TYPE_BOOLEAN; case PURPLE_ROOMLIST_FIELD_INT: case PURPLE_ROOMLIST_FIELD_STRING: types[j] = G_TYPE_STRING; model = gtk_tree_store_newv(columns, types); tree = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model)); selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree)); g_signal_connect(G_OBJECT(selection), "changed", G_CALLBACK(selection_changed_cb), grl); 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 (purple_roomlist_field_get_hidden(f)) renderer = gtk_cell_renderer_text_new(); column = gtk_tree_view_column_new_with_attributes( purple_roomlist_field_get_label(f), renderer, 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 (purple_roomlist_field_get_field_type(f) == 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); #if 0 /* uncomment this when the tooltips are slightly less annoying and more well behaved */ 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); pidgin_tooltip_setup_for_treeview(tree, list, pidgin_roomlist_create_tooltip, pidgin_roomlist_paint_tooltip); /* 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 = purple_roomlist_get_ui_data(list); if (!rl || !rl->dialog || !rl->dialog->pg_needs_pulse) { rl->dialog->pg_update_to = 0; gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress)); rl->dialog->pg_needs_pulse = FALSE; static void pidgin_roomlist_add_room(PurpleRoomlist *list, PurpleRoomlistRoom *room) PidginRoomlist *rl = purple_roomlist_get_ui_data(list); GtkTreeRowReference *rr, *parentrr = NULL; GtkTreeIter iter, parent, child; if (purple_roomlist_room_get_room_type(room) == PURPLE_ROOMLIST_ROOMTYPE_ROOM) if (rl->dialog->pg_update_to == 0) { rl->dialog->pg_update_to = g_timeout_add(100, pidgin_progress_bar_pulse, list); gtk_progress_bar_pulse(GTK_PROGRESS_BAR(rl->dialog->progress)); rl->dialog->pg_needs_pulse = TRUE; if (purple_roomlist_room_get_parent(room)) { parentrr = g_hash_table_lookup(rl->cats, purple_roomlist_room_get_parent(room)); path = gtk_tree_row_reference_get_path(parentrr); 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); gtk_tree_store_append(rl->model, &iter, (parentrr ? &parent : NULL)); if (purple_roomlist_room_get_room_type(room) & 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 (purple_roomlist_room_get_room_type(room) & 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, purple_roomlist_room_get_name(room), -1); gtk_tree_store_set(rl->model, &iter, ROOM_COLUMN, room, -1); l = purple_roomlist_room_get_fields(room), k = purple_roomlist_get_fields(list); l && k; j++, l = l->next, k = k->next) PurpleRoomlistField *f = k->data; if (purple_roomlist_field_get_hidden(f)) gtk_tree_store_set(rl->model, &iter, j, l->data, -1); static void pidgin_roomlist_in_progress(PurpleRoomlist *list, gboolean in_progress) PidginRoomlist *rl = purple_roomlist_get_ui_data(list); 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); 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 = purple_roomlist_get_ui_data(list); roomlists = g_list_remove(roomlists, list); g_return_if_fail(rl != NULL); g_hash_table_destroy(rl->cats); purple_roomlist_set_ui_data(list, NULL); static PurpleRoomlistUiOps ops = { pidgin_roomlist_dialog_show_with_account, pidgin_roomlist_set_fields, pidgin_roomlist_add_room, pidgin_roomlist_in_progress, void pidgin_roomlist_init(void) purple_roomlist_set_ui_ops(&ops);