pidgin/pidgin

1966704b3e42
merge of '77693555855fe9cd3215414f79964dba346cc5fa'
and '19a87e98e5857ad0289f2c760d460f7f1dbbb42d'
/*
* @file gtkblist.c GTK+ BuddyList API
* @ingroup gtkui
*
* gaim
*
* Gaim 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include "internal.h"
#include "gtkgaim.h"
#include "account.h"
#include "connection.h"
#include "core.h"
#include "debug.h"
#include "notify.h"
#include "prpl.h"
#include "prefs.h"
#include "plugin.h"
#include "request.h"
#include "signals.h"
#include "gaimstock.h"
#include "util.h"
#include "gtkaccount.h"
#include "gtkblist.h"
#include "gtkcellrendererexpander.h"
#include "gtkconv.h"
#include "gtkdebug.h"
#include "gtkdialogs.h"
#include "gtkft.h"
#include "gtklog.h"
#include "gtkmenutray.h"
#include "gtkpounce.h"
#include "gtkplugin.h"
#include "gtkprefs.h"
#include "gtkprivacy.h"
#include "gtkroomlist.h"
#include "gtkstatusbox.h"
#include "gtkscrollbook.h"
#include "gtkutils.h"
#include <gdk/gdkkeysyms.h>
#include <gtk/gtk.h>
#include <gdk/gdk.h>
#define HEADLINE_CLOSE_SIZE 12
typedef struct
{
GaimAccount *account;
GtkWidget *window;
GtkWidget *combo;
GtkWidget *entry;
GtkWidget *entry_for_alias;
GtkWidget *account_box;
} GaimGtkAddBuddyData;
typedef struct
{
GaimAccount *account;
gchar *default_chat_name;
GtkWidget *window;
GtkWidget *account_menu;
GtkWidget *alias_entry;
GtkWidget *group_combo;
GtkWidget *entries_box;
GtkSizeGroup *sg;
GList *entries;
} GaimGtkAddChatData;
typedef struct
{
GaimAccount *account;
GtkWidget *window;
GtkWidget *account_menu;
GtkWidget *entries_box;
GtkSizeGroup *sg;
GList *entries;
} GaimGtkJoinChatData;
static GtkWidget *accountmenu = NULL;
static guint visibility_manager_count = 0;
static gboolean gtk_blist_obscured = FALSE;
GHashTable* status_icon_hash_table = NULL;
static GList *gaim_gtk_blist_sort_methods = NULL;
static struct gaim_gtk_blist_sort_method *current_sort_method = NULL;
static void sort_method_none(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter);
/* The functions we use for sorting aren't available in gtk 2.0.x, and
* segfault in 2.2.0. 2.2.1 is known to work, so I'll require that */
#if GTK_CHECK_VERSION(2,2,1)
static void sort_method_alphabetical(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter);
static void sort_method_status(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter);
static void sort_method_log(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter);
#endif
static GaimGtkBuddyList *gtkblist = NULL;
static gboolean gaim_gtk_blist_refresh_timer(GaimBuddyList *list);
static void gaim_gtk_blist_update_buddy(GaimBuddyList *list, GaimBlistNode *node, gboolean statusChange);
static void gaim_gtk_blist_selection_changed(GtkTreeSelection *selection, gpointer data);
static void gaim_gtk_blist_update(GaimBuddyList *list, GaimBlistNode *node);
static void gaim_gtk_blist_update_contact(GaimBuddyList *list, GaimBlistNode *node);
static char *gaim_get_tooltip_text(GaimBlistNode *node, gboolean full);
static const char *item_factory_translate_func (const char *path, gpointer func_data);
static gboolean get_iter_from_node(GaimBlistNode *node, GtkTreeIter *iter);
static void redo_buddy_list(GaimBuddyList *list, gboolean remove, gboolean rerender);
static void gaim_gtk_blist_collapse_contact_cb(GtkWidget *w, GaimBlistNode *node);
static char *gaim_get_group_title(GaimBlistNode *gnode, gboolean expanded);
static void gaim_gtk_blist_tooltip_destroy(void);
struct _gaim_gtk_blist_node {
GtkTreeRowReference *row;
gboolean contact_expanded;
gboolean recent_signonoff;
gint recent_signonoff_timer;
GString *status_icon_key;
};
static void gaim_gtk_blist_update_buddy_status_icon_key(struct _gaim_gtk_blist_node *gtkbuddynode,
GaimBuddy *buddy, GaimStatusIconSize size);
static char dim_grey_string[8] = "";
static char *dim_grey()
{
if (!gtkblist)
return "dim grey";
if (!dim_grey_string[0]) {
GtkStyle *style = gtk_widget_get_style(gtkblist->treeview);
snprintf(dim_grey_string, sizeof(dim_grey_string), "#%02x%02x%02x",
style->text_aa[GTK_STATE_NORMAL].red >> 8,
style->text_aa[GTK_STATE_NORMAL].green >> 8,
style->text_aa[GTK_STATE_NORMAL].blue >> 8);
}
return dim_grey_string;
}
/***************************************************
* Callbacks *
***************************************************/
static gboolean gtk_blist_visibility_cb(GtkWidget *w, GdkEventVisibility *event, gpointer data)
{
if (event->state == GDK_VISIBILITY_FULLY_OBSCURED)
gtk_blist_obscured = TRUE;
else if (gtk_blist_obscured) {
gtk_blist_obscured = FALSE;
gaim_gtk_blist_refresh_timer(gaim_get_blist());
}
/* continue to handle event normally */
return FALSE;
}
static gboolean gtk_blist_window_state_cb(GtkWidget *w, GdkEventWindowState *event, gpointer data)
{
if(event->changed_mask & GDK_WINDOW_STATE_WITHDRAWN) {
if(event->new_window_state & GDK_WINDOW_STATE_WITHDRAWN)
gaim_prefs_set_bool("/gaim/gtk/blist/list_visible", FALSE);
else {
gaim_prefs_set_bool("/gaim/gtk/blist/list_visible", TRUE);
gaim_gtk_blist_refresh_timer(gaim_get_blist());
}
}
if(event->changed_mask & GDK_WINDOW_STATE_MAXIMIZED) {
if(event->new_window_state & GDK_WINDOW_STATE_MAXIMIZED)
gaim_prefs_set_bool("/gaim/gtk/blist/list_maximized", TRUE);
else
gaim_prefs_set_bool("/gaim/gtk/blist/list_maximized", FALSE);
}
/* Refresh gtkblist if un-iconifying */
if (event->changed_mask & GDK_WINDOW_STATE_ICONIFIED){
if (!(event->new_window_state & GDK_WINDOW_STATE_ICONIFIED))
gaim_gtk_blist_refresh_timer(gaim_get_blist());
}
return FALSE;
}
static gboolean gtk_blist_delete_cb(GtkWidget *w, GdkEventAny *event, gpointer data)
{
if(visibility_manager_count)
gaim_blist_set_visible(FALSE);
else
gaim_core_quit();
/* we handle everything, event should not propogate further */
return TRUE;
}
static gboolean gtk_blist_configure_cb(GtkWidget *w, GdkEventConfigure *event, gpointer data)
{
/* unfortunately GdkEventConfigure ignores the window gravity, but *
* the only way we have of setting the position doesn't. we have to *
* call get_position because it does pay attention to the gravity. *
* this is inefficient and I agree it sucks, but it's more likely *
* to work correctly. - Robot101 */
gint x, y;
/* check for visibility because when we aren't visible, this will *
* give us bogus (0,0) coordinates. - xOr */
if (GTK_WIDGET_VISIBLE(w))
gtk_window_get_position(GTK_WINDOW(w), &x, &y);
else
return FALSE; /* carry on normally */
#ifdef _WIN32
/* Workaround for GTK+ bug # 169811 - "configure_event" is fired
* when the window is being maximized */
if (gdk_window_get_state(w->window)
& GDK_WINDOW_STATE_MAXIMIZED) {
return FALSE;
}
#endif
/* don't save if nothing changed */
if (x == gaim_prefs_get_int("/gaim/gtk/blist/x") &&
y == gaim_prefs_get_int("/gaim/gtk/blist/y") &&
event->width == gaim_prefs_get_int("/gaim/gtk/blist/width") &&
event->height == gaim_prefs_get_int("/gaim/gtk/blist/height")) {
return FALSE; /* carry on normally */
}
/* don't save off-screen positioning */
if (x + event->width < 0 ||
y + event->height < 0 ||
x > gdk_screen_width() ||
y > gdk_screen_height()) {
return FALSE; /* carry on normally */
}
/* ignore changes when maximized */
if(gaim_prefs_get_bool("/gaim/gtk/blist/list_maximized"))
return FALSE;
/* store the position */
gaim_prefs_set_int("/gaim/gtk/blist/x", x);
gaim_prefs_set_int("/gaim/gtk/blist/y", y);
gaim_prefs_set_int("/gaim/gtk/blist/width", event->width);
gaim_prefs_set_int("/gaim/gtk/blist/height", event->height);
gtk_widget_set_size_request(gtkblist->headline_label,
gaim_prefs_get_int("/gaim/gtk/blist/width")-25,-1);
/* continue to handle event normally */
return FALSE;
}
static void gtk_blist_menu_info_cb(GtkWidget *w, GaimBuddy *b)
{
serv_get_info(b->account->gc, b->name);
}
static void gtk_blist_menu_im_cb(GtkWidget *w, GaimBuddy *b)
{
gaim_gtkdialogs_im_with_user(b->account, b->name);
}
static void gtk_blist_menu_send_file_cb(GtkWidget *w, GaimBuddy *b)
{
serv_send_file(b->account->gc, b->name, NULL);
}
static void gtk_blist_menu_autojoin_cb(GtkWidget *w, GaimChat *chat)
{
gaim_blist_node_set_bool((GaimBlistNode*)chat, "gtk-autojoin",
gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(w)));
}
static void gtk_blist_join_chat(GaimChat *chat)
{
GaimConversation *conv;
conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_CHAT,
gaim_chat_get_name(chat),
chat->account);
if (conv != NULL)
gaim_gtkconv_present_conversation(conv);
serv_join_chat(chat->account->gc, chat->components);
}
static void gtk_blist_menu_join_cb(GtkWidget *w, GaimChat *chat)
{
gtk_blist_join_chat(chat);
}
static void gtk_blist_renderer_edited_cb(GtkCellRendererText *text_rend, char *arg1,
char *arg2, gpointer nada)
{
GtkTreeIter iter;
GtkTreePath *path;
GValue val;
GaimBlistNode *node;
GaimGroup *dest;
path = gtk_tree_path_new_from_string (arg1);
gtk_tree_model_get_iter (GTK_TREE_MODEL(gtkblist->treemodel), &iter, path);
gtk_tree_path_free (path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
gtk_tree_view_set_enable_search (GTK_TREE_VIEW(gtkblist->treeview), TRUE);
g_object_set(G_OBJECT(gtkblist->text_rend), "editable", FALSE, NULL);
switch (node->type)
{
case GAIM_BLIST_CONTACT_NODE:
{
GaimContact *contact = (GaimContact *)node;
struct _gaim_gtk_blist_node *gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
if (contact->alias || gtknode->contact_expanded)
gaim_blist_alias_contact(contact, arg2);
else
{
GaimBuddy *buddy = gaim_contact_get_priority_buddy(contact);
gaim_blist_alias_buddy(buddy, arg2);
serv_alias_buddy(buddy);
}
}
break;
case GAIM_BLIST_BUDDY_NODE:
gaim_blist_alias_buddy((GaimBuddy*)node, arg2);
serv_alias_buddy((GaimBuddy *)node);
break;
case GAIM_BLIST_GROUP_NODE:
dest = gaim_find_group(arg2);
if (dest != NULL && strcmp(arg2, ((GaimGroup*) node)->name)) {
gaim_gtkdialogs_merge_groups((GaimGroup*) node, arg2);
} else
gaim_blist_rename_group((GaimGroup*)node, arg2);
break;
case GAIM_BLIST_CHAT_NODE:
gaim_blist_alias_chat((GaimChat*)node, arg2);
break;
default:
break;
}
}
static void gtk_blist_menu_alias_cb(GtkWidget *w, GaimBlistNode *node)
{
GtkTreeIter iter;
GtkTreePath *path;
const char *text = NULL;
char *esc;
if (!(get_iter_from_node(node, &iter))) {
/* This is either a bug, or the buddy is in a collapsed contact */
node = node->parent;
if (!get_iter_from_node(node, &iter))
/* Now it's definitely a bug */
return;
}
switch (node->type) {
case GAIM_BLIST_BUDDY_NODE:
text = gaim_buddy_get_alias((GaimBuddy *)node);
break;
case GAIM_BLIST_CONTACT_NODE:
text = gaim_contact_get_alias((GaimContact *)node);
break;
case GAIM_BLIST_GROUP_NODE:
text = ((GaimGroup *)node)->name;
break;
case GAIM_BLIST_CHAT_NODE:
text = gaim_chat_get_name((GaimChat *)node);
break;
default:
g_return_if_reached();
}
esc = g_markup_escape_text(text, -1);
gtk_tree_store_set(gtkblist->treemodel, &iter, NAME_COLUMN, esc, -1);
g_free(esc);
path = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel), &iter);
g_object_set(G_OBJECT(gtkblist->text_rend), "editable", TRUE, NULL);
gtk_tree_view_set_enable_search (GTK_TREE_VIEW(gtkblist->treeview), FALSE);
gtk_widget_grab_focus(gtkblist->treeview);
#if GTK_CHECK_VERSION(2,2,0)
gtk_tree_view_set_cursor_on_cell(GTK_TREE_VIEW(gtkblist->treeview), path,
gtkblist->text_column, gtkblist->text_rend, TRUE);
#else
gtk_tree_view_set_cursor(GTK_TREE_VIEW(gtkblist->treeview), path, gtkblist->text_column, TRUE);
#endif
gtk_tree_path_free(path);
}
static void gtk_blist_menu_bp_cb(GtkWidget *w, GaimBuddy *b)
{
gaim_gtk_pounce_editor_show(b->account, b->name, NULL);
}
static void gtk_blist_menu_showlog_cb(GtkWidget *w, GaimBlistNode *node)
{
GaimLogType type;
GaimAccount *account;
char *name = NULL;
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
if (GAIM_BLIST_NODE_IS_BUDDY(node)) {
GaimBuddy *b = (GaimBuddy*) node;
type = GAIM_LOG_IM;
name = g_strdup(b->name);
account = b->account;
} else if (GAIM_BLIST_NODE_IS_CHAT(node)) {
GaimChat *c = (GaimChat*) node;
GaimPluginProtocolInfo *prpl_info = NULL;
type = GAIM_LOG_CHAT;
account = c->account;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gaim_find_prpl(gaim_account_get_protocol_id(account)));
if (prpl_info && prpl_info->get_chat_name) {
name = prpl_info->get_chat_name(c->components);
}
} else if (GAIM_BLIST_NODE_IS_CONTACT(node)) {
gaim_gtk_log_show_contact((GaimContact *)node);
gaim_gtk_clear_cursor(gtkblist->window);
return;
} else {
gaim_gtk_clear_cursor(gtkblist->window);
/* This callback should not have been registered for a node
* that doesn't match the type of one of the blocks above. */
g_return_if_reached();
}
if (name && account) {
gaim_gtk_log_show(type, name, account);
g_free(name);
gaim_gtk_clear_cursor(gtkblist->window);
}
}
static void gtk_blist_show_systemlog_cb()
{
gaim_gtk_syslog_show();
}
static void gtk_blist_show_onlinehelp_cb()
{
gaim_notify_uri(NULL, GAIM_WEBSITE "documentation.php");
}
static void
do_join_chat(GaimGtkJoinChatData *data)
{
if (data)
{
GHashTable *components =
g_hash_table_new_full(g_str_hash, g_str_equal, g_free, g_free);
GList *tmp;
GaimChat *chat;
for (tmp = data->entries; tmp != NULL; tmp = tmp->next)
{
if (g_object_get_data(tmp->data, "is_spin"))
{
g_hash_table_replace(components,
g_strdup(g_object_get_data(tmp->data, "identifier")),
g_strdup_printf("%d",
gtk_spin_button_get_value_as_int(tmp->data)));
}
else
{
g_hash_table_replace(components,
g_strdup(g_object_get_data(tmp->data, "identifier")),
g_strdup(gtk_entry_get_text(tmp->data)));
}
}
chat = gaim_chat_new(data->account, NULL, components);
gtk_blist_join_chat(chat);
gaim_blist_remove_chat(chat);
}
}
static void
do_joinchat(GtkWidget *dialog, int id, GaimGtkJoinChatData *info)
{
switch(id)
{
case GTK_RESPONSE_OK:
do_join_chat(info);
break;
}
gtk_widget_destroy(GTK_WIDGET(dialog));
g_list_free(info->entries);
g_free(info);
}
/*
* Check the values of all the text entry boxes. If any required input
* strings are empty then don't allow the user to click on "OK."
*/
static void
joinchat_set_sensitive_if_input_cb(GtkWidget *entry, gpointer user_data)
{
GaimGtkJoinChatData *data;
GList *tmp;
const char *text;
gboolean required;
gboolean sensitive = TRUE;
data = user_data;
for (tmp = data->entries; tmp != NULL; tmp = tmp->next)
{
if (!g_object_get_data(tmp->data, "is_spin"))
{
required = GPOINTER_TO_INT(g_object_get_data(tmp->data, "required"));
text = gtk_entry_get_text(tmp->data);
if (required && (*text == '\0'))
sensitive = FALSE;
}
}
gtk_dialog_set_response_sensitive(GTK_DIALOG(data->window), GTK_RESPONSE_OK, sensitive);
}
static void
gaim_gtk_blist_update_privacy_cb(GaimBuddy *buddy)
{
gaim_gtk_blist_update_buddy(gaim_get_blist(), (GaimBlistNode*)(buddy), TRUE);
}
static void
rebuild_joinchat_entries(GaimGtkJoinChatData *data)
{
GaimConnection *gc;
GList *list = NULL, *tmp;
GHashTable *defaults = NULL;
struct proto_chat_entry *pce;
gboolean focus = TRUE;
g_return_if_fail(data->account != NULL);
gc = gaim_account_get_connection(data->account);
while ((tmp = gtk_container_get_children(GTK_CONTAINER(data->entries_box))))
gtk_widget_destroy(tmp->data);
g_list_free(data->entries);
data->entries = NULL;
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info != NULL)
list = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info(gc);
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info_defaults != NULL)
defaults = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info_defaults(gc, NULL);
for (tmp = list; tmp; tmp = tmp->next)
{
GtkWidget *label;
GtkWidget *rowbox;
GtkWidget *input;
pce = tmp->data;
rowbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
gtk_box_pack_start(GTK_BOX(data->entries_box), rowbox, FALSE, FALSE, 0);
label = gtk_label_new_with_mnemonic(pce->label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_size_group_add_widget(data->sg, label);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
if (pce->is_int)
{
GtkObject *adjust;
adjust = gtk_adjustment_new(pce->min, pce->min, pce->max,
1, 10, 10);
input = gtk_spin_button_new(GTK_ADJUSTMENT(adjust), 1, 0);
gtk_widget_set_size_request(input, 50, -1);
gtk_box_pack_end(GTK_BOX(rowbox), input, FALSE, FALSE, 0);
}
else
{
char *value;
input = gtk_entry_new();
gtk_entry_set_activates_default(GTK_ENTRY(input), TRUE);
value = g_hash_table_lookup(defaults, pce->identifier);
if (value != NULL)
gtk_entry_set_text(GTK_ENTRY(input), value);
if (pce->secret)
{
gtk_entry_set_visibility(GTK_ENTRY(input), FALSE);
if (gtk_entry_get_invisible_char(GTK_ENTRY(input)) == '*')
gtk_entry_set_invisible_char(GTK_ENTRY(input), GAIM_INVISIBLE_CHAR);
}
gtk_box_pack_end(GTK_BOX(rowbox), input, TRUE, TRUE, 0);
g_signal_connect(G_OBJECT(input), "changed",
G_CALLBACK(joinchat_set_sensitive_if_input_cb), data);
}
/* Do the following for any type of input widget */
if (focus)
{
gtk_widget_grab_focus(input);
focus = FALSE;
}
gtk_label_set_mnemonic_widget(GTK_LABEL(label), input);
gaim_set_accessible_label(input, label);
g_object_set_data(G_OBJECT(input), "identifier", (gpointer)pce->identifier);
g_object_set_data(G_OBJECT(input), "is_spin", GINT_TO_POINTER(pce->is_int));
g_object_set_data(G_OBJECT(input), "required", GINT_TO_POINTER(pce->required));
data->entries = g_list_append(data->entries, input);
g_free(pce);
}
g_list_free(list);
g_hash_table_destroy(defaults);
/* Set whether the "OK" button should be clickable initially */
joinchat_set_sensitive_if_input_cb(NULL, data);
gtk_widget_show_all(data->entries_box);
}
static void
joinchat_select_account_cb(GObject *w, GaimAccount *account,
GaimGtkJoinChatData *data)
{
data->account = account;
rebuild_joinchat_entries(data);
}
static gboolean
chat_account_filter_func(GaimAccount *account)
{
GaimConnection *gc = gaim_account_get_connection(account);
GaimPluginProtocolInfo *prpl_info = NULL;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
return (prpl_info->chat_info != NULL);
}
gboolean
gaim_gtk_blist_joinchat_is_showable()
{
GList *c;
GaimConnection *gc;
for (c = gaim_connections_get_all(); c != NULL; c = c->next) {
gc = c->data;
if (chat_account_filter_func(gaim_connection_get_account(gc)))
return TRUE;
}
return FALSE;
}
void
gaim_gtk_blist_joinchat_show(void)
{
GtkWidget *hbox, *vbox;
GtkWidget *rowbox;
GtkWidget *label;
GaimGtkBuddyList *gtkblist;
GtkWidget *img = NULL;
GaimGtkJoinChatData *data = NULL;
gtkblist = GAIM_GTK_BLIST(gaim_get_blist());
img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_QUESTION,
GTK_ICON_SIZE_DIALOG);
data = g_new0(GaimGtkJoinChatData, 1);
data->window = gtk_dialog_new_with_buttons(_("Join a Chat"),
NULL, GTK_DIALOG_NO_SEPARATOR,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GAIM_STOCK_CHAT, GTK_RESPONSE_OK, NULL);
gtk_dialog_set_default_response(GTK_DIALOG(data->window), GTK_RESPONSE_OK);
gtk_container_set_border_width(GTK_CONTAINER(data->window), GAIM_HIG_BOX_SPACE);
gtk_window_set_resizable(GTK_WINDOW(data->window), FALSE);
gtk_box_set_spacing(GTK_BOX(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BORDER);
gtk_container_set_border_width(
GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BOX_SPACE);
gtk_window_set_role(GTK_WINDOW(data->window), "join_chat");
hbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
gtk_container_add(GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), hbox);
gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
vbox = gtk_vbox_new(FALSE, 5);
gtk_container_set_border_width(GTK_CONTAINER(vbox), 0);
gtk_container_add(GTK_CONTAINER(hbox), vbox);
label = gtk_label_new(_("Please enter the appropriate information "
"about the chat you would like to join.\n"));
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
rowbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
gtk_box_pack_start(GTK_BOX(vbox), rowbox, TRUE, TRUE, 0);
data->sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
label = gtk_label_new_with_mnemonic(_("_Account:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
gtk_size_group_add_widget(data->sg, label);
data->account_menu = gaim_gtk_account_option_menu_new(NULL, FALSE,
G_CALLBACK(joinchat_select_account_cb),
chat_account_filter_func, data);
gtk_box_pack_start(GTK_BOX(rowbox), data->account_menu, TRUE, TRUE, 0);
gtk_label_set_mnemonic_widget(GTK_LABEL(label),
GTK_WIDGET(data->account_menu));
gaim_set_accessible_label (data->account_menu, label);
data->entries_box = gtk_vbox_new(FALSE, 5);
gtk_container_add(GTK_CONTAINER(vbox), data->entries_box);
gtk_container_set_border_width(GTK_CONTAINER(data->entries_box), 0);
data->account = gaim_gtk_account_option_menu_get_selected(data->account_menu);
rebuild_joinchat_entries(data);
g_signal_connect(G_OBJECT(data->window), "response",
G_CALLBACK(do_joinchat), data);
g_object_unref(data->sg);
gtk_widget_show_all(data->window);
}
static void gtk_blist_row_expanded_cb(GtkTreeView *tv, GtkTreeIter *iter, GtkTreePath *path, gpointer user_data) {
GaimBlistNode *node;
GValue val;
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel), iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_GROUP(node)) {
char *title;
title = gaim_get_group_title(node, TRUE);
gtk_tree_store_set(gtkblist->treemodel, iter,
NAME_COLUMN, title,
-1);
g_free(title);
gaim_blist_node_set_bool(node, "collapsed", FALSE);
}
}
static void gtk_blist_row_collapsed_cb(GtkTreeView *tv, GtkTreeIter *iter, GtkTreePath *path, gpointer user_data) {
GaimBlistNode *node;
GValue val;
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel), iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_GROUP(node)) {
char *title;
title = gaim_get_group_title(node, FALSE);
gtk_tree_store_set(gtkblist->treemodel, iter,
NAME_COLUMN, title,
-1);
g_free(title);
gaim_blist_node_set_bool(node, "collapsed", TRUE);
} else if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
gaim_gtk_blist_collapse_contact_cb(NULL, node);
}
}
static void gtk_blist_row_activated_cb(GtkTreeView *tv, GtkTreePath *path, GtkTreeViewColumn *col, gpointer data) {
GaimBlistNode *node;
GtkTreeIter iter;
GValue val;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if(GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_BUDDY(node)) {
GaimBuddy *buddy;
if(GAIM_BLIST_NODE_IS_CONTACT(node))
buddy = gaim_contact_get_priority_buddy((GaimContact*)node);
else
buddy = (GaimBuddy*)node;
gaim_gtkdialogs_im_with_user(buddy->account, buddy->name);
} else if (GAIM_BLIST_NODE_IS_CHAT(node)) {
gtk_blist_join_chat((GaimChat *)node);
} else if (GAIM_BLIST_NODE_IS_GROUP(node)) {
/* if (gtk_tree_view_row_expanded(tv, path))
gtk_tree_view_collapse_row(tv, path);
else
gtk_tree_view_expand_row(tv,path,FALSE);*/
}
}
static void gaim_gtk_blist_add_chat_cb()
{
GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkblist->treeview));
GtkTreeIter iter;
GaimBlistNode *node;
if(gtk_tree_selection_get_selected(sel, NULL, &iter)){
gtk_tree_model_get(GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &node, -1);
if (GAIM_BLIST_NODE_IS_BUDDY(node))
gaim_blist_request_add_chat(NULL, (GaimGroup*)node->parent->parent, NULL, NULL);
if (GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_CHAT(node))
gaim_blist_request_add_chat(NULL, (GaimGroup*)node->parent, NULL, NULL);
else if (GAIM_BLIST_NODE_IS_GROUP(node))
gaim_blist_request_add_chat(NULL, (GaimGroup*)node, NULL, NULL);
}
else {
gaim_blist_request_add_chat(NULL, NULL, NULL, NULL);
}
}
static void gaim_gtk_blist_add_buddy_cb()
{
GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkblist->treeview));
GtkTreeIter iter;
GaimBlistNode *node;
if(gtk_tree_selection_get_selected(sel, NULL, &iter)){
gtk_tree_model_get(GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &node, -1);
if (GAIM_BLIST_NODE_IS_BUDDY(node)) {
gaim_blist_request_add_buddy(NULL, NULL, ((GaimGroup*)node->parent->parent)->name,
NULL);
} else if (GAIM_BLIST_NODE_IS_CONTACT(node)
|| GAIM_BLIST_NODE_IS_CHAT(node)) {
gaim_blist_request_add_buddy(NULL, NULL, ((GaimGroup*)node->parent)->name, NULL);
} else if (GAIM_BLIST_NODE_IS_GROUP(node)) {
gaim_blist_request_add_buddy(NULL, NULL, ((GaimGroup*)node)->name, NULL);
}
}
else {
gaim_blist_request_add_buddy(NULL, NULL, NULL, NULL);
}
}
static void
gaim_gtk_blist_remove_cb (GtkWidget *w, GaimBlistNode *node)
{
if (GAIM_BLIST_NODE_IS_BUDDY(node)) {
gaim_gtkdialogs_remove_buddy((GaimBuddy*)node);
} else if (GAIM_BLIST_NODE_IS_CHAT(node)) {
gaim_gtkdialogs_remove_chat((GaimChat*)node);
} else if (GAIM_BLIST_NODE_IS_GROUP(node)) {
gaim_gtkdialogs_remove_group((GaimGroup*)node);
} else if (GAIM_BLIST_NODE_IS_CONTACT(node)) {
gaim_gtkdialogs_remove_contact((GaimContact*)node);
}
}
struct _expand {
GtkTreeView *treeview;
GtkTreePath *path;
GaimBlistNode *node;
};
static gboolean
scroll_to_expanded_cell(gpointer data)
{
struct _expand *ex = data;
gtk_tree_view_scroll_to_cell(ex->treeview, ex->path, NULL, FALSE, 0, 0);
gaim_gtk_blist_update_contact(NULL, ex->node);
gtk_tree_path_free(ex->path);
g_free(ex);
return FALSE;
}
static void
gaim_gtk_blist_expand_contact_cb(GtkWidget *w, GaimBlistNode *node)
{
struct _gaim_gtk_blist_node *gtknode;
GtkTreeIter iter, parent;
GaimBlistNode *bnode;
GtkTreePath *path;
if(!GAIM_BLIST_NODE_IS_CONTACT(node))
return;
gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
gtknode->contact_expanded = TRUE;
for(bnode = node->child; bnode; bnode = bnode->next) {
gaim_gtk_blist_update(NULL, bnode);
}
/* This ensures that the bottom buddy is visible, i.e. not scrolled off the alignment */
if (get_iter_from_node(node, &parent)) {
struct _expand *ex = g_new0(struct _expand, 1);
gtk_tree_model_iter_nth_child(GTK_TREE_MODEL(gtkblist->treemodel), &iter, &parent,
gtk_tree_model_iter_n_children(GTK_TREE_MODEL(gtkblist->treemodel), &parent) -1);
path = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel), &iter);
/* Let the treeview draw so it knows where to scroll */
ex->treeview = GTK_TREE_VIEW(gtkblist->treeview);
ex->path = path;
ex->node = node->child;
g_idle_add(scroll_to_expanded_cell, ex);
}
}
static void
gaim_gtk_blist_collapse_contact_cb(GtkWidget *w, GaimBlistNode *node)
{
GaimBlistNode *bnode;
struct _gaim_gtk_blist_node *gtknode;
if(!GAIM_BLIST_NODE_IS_CONTACT(node))
return;
gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
gtknode->contact_expanded = FALSE;
for(bnode = node->child; bnode; bnode = bnode->next) {
gaim_gtk_blist_update(NULL, bnode);
}
}
static void
toggle_privacy(GtkWidget *widget, GaimBlistNode *node)
{
GaimBuddy *buddy;
GaimAccount *account;
gboolean permitted;
const char *name;
if (!GAIM_BLIST_NODE_IS_BUDDY(node))
return;
buddy = (GaimBuddy *)node;
account = gaim_buddy_get_account(buddy);
name = gaim_buddy_get_name(buddy);
permitted = gaim_privacy_check(account, name);
/* XXX: Perhaps ask whether to restore the previous lists where appropirate? */
if (permitted)
gaim_privacy_deny(account, name, FALSE, FALSE);
else
gaim_privacy_allow(account, name, FALSE, FALSE);
gaim_gtk_blist_update(gaim_get_blist(), node);
}
void gaim_gtk_append_blist_node_privacy_menu(GtkWidget *menu, GaimBlistNode *node)
{
GaimBuddy *buddy = (GaimBuddy *)node;
GaimAccount *account;
gboolean permitted;
account = gaim_buddy_get_account(buddy);
permitted = gaim_privacy_check(account, gaim_buddy_get_name(buddy));
gaim_new_item_from_stock(menu, permitted ? _("_Block") : _("Un_block"),
GTK_STOCK_DIALOG_ERROR, G_CALLBACK(toggle_privacy),
node, 0 ,0, NULL);
}
void
gaim_gtk_append_blist_node_proto_menu(GtkWidget *menu, GaimConnection *gc,
GaimBlistNode *node)
{
GList *l, *ll;
GaimPluginProtocolInfo *prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
if(!prpl_info || !prpl_info->blist_node_menu)
return;
for(l = ll = prpl_info->blist_node_menu(node); l; l = l->next) {
GaimMenuAction *act = (GaimMenuAction *) l->data;
gaim_gtk_append_menu_action(menu, act, node);
}
g_list_free(ll);
}
void
gaim_gtk_append_blist_node_extended_menu(GtkWidget *menu, GaimBlistNode *node)
{
GList *l, *ll;
for(l = ll = gaim_blist_node_get_extended_menu(node); l; l = l->next) {
GaimMenuAction *act = (GaimMenuAction *) l->data;
gaim_gtk_append_menu_action(menu, act, node);
}
g_list_free(ll);
}
void
gaim_gtk_blist_make_buddy_menu(GtkWidget *menu, GaimBuddy *buddy, gboolean sub) {
GaimPluginProtocolInfo *prpl_info;
GaimContact *contact;
gboolean contact_expanded = FALSE;
g_return_if_fail(menu);
g_return_if_fail(buddy);
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(buddy->account->gc->prpl);
contact = gaim_buddy_get_contact(buddy);
if (contact) {
contact_expanded = ((struct _gaim_gtk_blist_node *)(((GaimBlistNode*)contact)->ui_data))->contact_expanded;
}
if (prpl_info && prpl_info->get_info) {
gaim_new_item_from_stock(menu, _("Get _Info"), GAIM_STOCK_INFO,
G_CALLBACK(gtk_blist_menu_info_cb), buddy, 0, 0, NULL);
}
gaim_new_item_from_stock(menu, _("I_M"), GAIM_STOCK_IM,
G_CALLBACK(gtk_blist_menu_im_cb), buddy, 0, 0, NULL);
if (prpl_info && prpl_info->send_file) {
if (!prpl_info->can_receive_file ||
prpl_info->can_receive_file(buddy->account->gc, buddy->name))
{
gaim_new_item_from_stock(menu, _("_Send File"),
GAIM_STOCK_FILE_TRANSFER,
G_CALLBACK(gtk_blist_menu_send_file_cb),
buddy, 0, 0, NULL);
}
}
gaim_new_item_from_stock(menu, _("Add Buddy _Pounce"), GAIM_STOCK_POUNCE,
G_CALLBACK(gtk_blist_menu_bp_cb), buddy, 0, 0, NULL);
if(((GaimBlistNode*)buddy)->parent->child->next && !sub && !contact_expanded) {
gaim_new_item_from_stock(menu, _("View _Log"), GAIM_STOCK_LOG,
G_CALLBACK(gtk_blist_menu_showlog_cb),
contact, 0, 0, NULL);
} else if (!sub) {
gaim_new_item_from_stock(menu, _("View _Log"), GAIM_STOCK_LOG,
G_CALLBACK(gtk_blist_menu_showlog_cb), buddy, 0, 0, NULL);
}
gaim_gtk_append_blist_node_privacy_menu(menu, (GaimBlistNode *)buddy);
gaim_gtk_append_blist_node_proto_menu(menu, buddy->account->gc,
(GaimBlistNode *)buddy);
gaim_gtk_append_blist_node_extended_menu(menu, (GaimBlistNode *)buddy);
if (((GaimBlistNode*)buddy)->parent->child->next && !sub && !contact_expanded) {
gaim_separator(menu);
gaim_new_item_from_stock(menu, _("Alias..."), GAIM_STOCK_ALIAS,
G_CALLBACK(gtk_blist_menu_alias_cb),
contact, 0, 0, NULL);
gaim_new_item_from_stock(menu, _("Remove"), GTK_STOCK_REMOVE,
G_CALLBACK(gaim_gtk_blist_remove_cb),
contact, 0, 0, NULL);
} else if (!sub || contact_expanded) {
gaim_separator(menu);
gaim_new_item_from_stock(menu, _("_Alias..."), GAIM_STOCK_ALIAS,
G_CALLBACK(gtk_blist_menu_alias_cb), buddy, 0, 0, NULL);
gaim_new_item_from_stock(menu, _("_Remove"), GTK_STOCK_REMOVE,
G_CALLBACK(gaim_gtk_blist_remove_cb), buddy,
0, 0, NULL);
}
}
static gboolean
gtk_blist_key_press_cb(GtkWidget *tv, GdkEventKey *event, gpointer data) {
GaimBlistNode *node;
GValue val;
GtkTreeIter iter;
GtkTreeSelection *sel;
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv));
if(!gtk_tree_selection_get_selected(sel, NULL, &iter))
return FALSE;
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel), &iter,
NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if(event->state & GDK_CONTROL_MASK &&
(event->keyval == 'o' || event->keyval == 'O')) {
GaimBuddy *buddy;
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
buddy = gaim_contact_get_priority_buddy((GaimContact*)node);
} else if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
buddy = (GaimBuddy*)node;
} else {
return FALSE;
}
if(buddy)
serv_get_info(buddy->account->gc, buddy->name);
}
return FALSE;
}
static GtkWidget *
create_group_menu (GaimBlistNode *node, GaimGroup *g)
{
GtkWidget *menu;
GtkWidget *item;
menu = gtk_menu_new();
gaim_new_item_from_stock(menu, _("Add a _Buddy"), GTK_STOCK_ADD,
G_CALLBACK(gaim_gtk_blist_add_buddy_cb), node, 0, 0, NULL);
item = gaim_new_item_from_stock(menu, _("Add a C_hat"), GTK_STOCK_ADD,
G_CALLBACK(gaim_gtk_blist_add_chat_cb), node, 0, 0, NULL);
gtk_widget_set_sensitive(item, gaim_gtk_blist_joinchat_is_showable());
gaim_new_item_from_stock(menu, _("_Delete Group"), GTK_STOCK_REMOVE,
G_CALLBACK(gaim_gtk_blist_remove_cb), node, 0, 0, NULL);
gaim_new_item_from_stock(menu, _("_Rename"), NULL,
G_CALLBACK(gtk_blist_menu_alias_cb), node, 0, 0, NULL);
gaim_gtk_append_blist_node_extended_menu(menu, node);
return menu;
}
static GtkWidget *
create_chat_menu(GaimBlistNode *node, GaimChat *c) {
GtkWidget *menu;
gboolean autojoin;
menu = gtk_menu_new();
autojoin = (gaim_blist_node_get_bool(node, "gtk-autojoin") ||
(gaim_blist_node_get_string(node, "gtk-autojoin") != NULL));
gaim_new_item_from_stock(menu, _("_Join"), GAIM_STOCK_CHAT,
G_CALLBACK(gtk_blist_menu_join_cb), node, 0, 0, NULL);
gaim_new_check_item(menu, _("Auto-Join"),
G_CALLBACK(gtk_blist_menu_autojoin_cb), node, autojoin);
gaim_new_item_from_stock(menu, _("View _Log"), GAIM_STOCK_LOG,
G_CALLBACK(gtk_blist_menu_showlog_cb), node, 0, 0, NULL);
gaim_gtk_append_blist_node_proto_menu(menu, c->account->gc, node);
gaim_gtk_append_blist_node_extended_menu(menu, node);
gaim_separator(menu);
gaim_new_item_from_stock(menu, _("_Alias..."), GAIM_STOCK_ALIAS,
G_CALLBACK(gtk_blist_menu_alias_cb), node, 0, 0, NULL);
gaim_new_item_from_stock(menu, _("_Remove"), GTK_STOCK_REMOVE,
G_CALLBACK(gaim_gtk_blist_remove_cb), node, 0, 0, NULL);
return menu;
}
static GtkWidget *
create_contact_menu (GaimBlistNode *node)
{
GtkWidget *menu;
menu = gtk_menu_new();
gaim_new_item_from_stock(menu, _("View _Log"), GAIM_STOCK_LOG,
G_CALLBACK(gtk_blist_menu_showlog_cb),
node, 0, 0, NULL);
gaim_separator(menu);
gaim_new_item_from_stock(menu, _("_Alias..."), GAIM_STOCK_ALIAS,
G_CALLBACK(gtk_blist_menu_alias_cb), node, 0, 0, NULL);
gaim_new_item_from_stock(menu, _("_Remove"), GTK_STOCK_REMOVE,
G_CALLBACK(gaim_gtk_blist_remove_cb), node, 0, 0, NULL);
gaim_separator(menu);
gaim_new_item_from_stock(menu, _("_Collapse"), GTK_STOCK_ZOOM_OUT,
G_CALLBACK(gaim_gtk_blist_collapse_contact_cb),
node, 0, 0, NULL);
gaim_gtk_append_blist_node_extended_menu(menu, node);
return menu;
}
static GtkWidget *
create_buddy_menu(GaimBlistNode *node, GaimBuddy *b) {
struct _gaim_gtk_blist_node *gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
GtkWidget *menu;
GtkWidget *menuitem;
gboolean show_offline = gaim_prefs_get_bool("/gaim/gtk/blist/show_offline_buddies");
menu = gtk_menu_new();
gaim_gtk_blist_make_buddy_menu(menu, b, FALSE);
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
gaim_separator(menu);
if(gtknode->contact_expanded) {
gaim_new_item_from_stock(menu, _("_Collapse"),
GTK_STOCK_ZOOM_OUT,
G_CALLBACK(gaim_gtk_blist_collapse_contact_cb),
node, 0, 0, NULL);
} else {
gaim_new_item_from_stock(menu, _("_Expand"),
GTK_STOCK_ZOOM_IN,
G_CALLBACK(gaim_gtk_blist_expand_contact_cb), node,
0, 0, NULL);
}
if(node->child->next) {
GaimBlistNode *bnode;
for(bnode = node->child; bnode; bnode = bnode->next) {
GaimBuddy *buddy = (GaimBuddy*)bnode;
GdkPixbuf *buf;
GtkWidget *submenu;
GtkWidget *image;
if(buddy == b)
continue;
if(!buddy->account->gc)
continue;
if(!show_offline && !GAIM_BUDDY_IS_ONLINE(buddy))
continue;
menuitem = gtk_image_menu_item_new_with_label(buddy->name);
buf = gaim_gtk_blist_get_status_icon(bnode,
GAIM_STATUS_ICON_SMALL);
image = gtk_image_new_from_pixbuf(buf);
g_object_unref(G_OBJECT(buf));
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem),
image);
gtk_widget_show(image);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
gtk_widget_show(menuitem);
submenu = gtk_menu_new();
gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), submenu);
gtk_widget_show(submenu);
gaim_gtk_blist_make_buddy_menu(submenu, buddy, TRUE);
}
}
}
return menu;
}
static gboolean
gaim_gtk_blist_show_context_menu(GaimBlistNode *node,
GtkMenuPositionFunc func,
GtkWidget *tv,
guint button,
guint32 time)
{
struct _gaim_gtk_blist_node *gtknode;
GtkWidget *menu = NULL;
gboolean handled = FALSE;
gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
/* Create a menu based on the thing we right-clicked on */
if (GAIM_BLIST_NODE_IS_GROUP(node)) {
GaimGroup *g = (GaimGroup *)node;
menu = create_group_menu(node, g);
} else if (GAIM_BLIST_NODE_IS_CHAT(node)) {
GaimChat *c = (GaimChat *)node;
menu = create_chat_menu(node, c);
} else if ((GAIM_BLIST_NODE_IS_CONTACT(node)) && (gtknode->contact_expanded)) {
menu = create_contact_menu(node);
} else if (GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_BUDDY(node)) {
GaimBuddy *b;
if (GAIM_BLIST_NODE_IS_CONTACT(node))
b = gaim_contact_get_priority_buddy((GaimContact*)node);
else
b = (GaimBuddy *)node;
menu = create_buddy_menu(node, b);
}
#ifdef _WIN32
/* Unhook the tooltip-timeout since we don't want a tooltip
* to appear and obscure the context menu we are about to show
This is a workaround for GTK+ bug 107320. */
if (gtkblist->timeout) {
g_source_remove(gtkblist->timeout);
gtkblist->timeout = 0;
}
#endif
/* Now display the menu */
if (menu != NULL) {
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, func, tv, button, time);
handled = TRUE;
}
return handled;
}
static gboolean gtk_blist_button_press_cb(GtkWidget *tv, GdkEventButton *event, gpointer user_data)
{
GtkTreePath *path;
GaimBlistNode *node;
GValue val;
GtkTreeIter iter;
GtkTreeSelection *sel;
GaimPlugin *prpl = NULL;
GaimPluginProtocolInfo *prpl_info = NULL;
struct _gaim_gtk_blist_node *gtknode;
gboolean handled = FALSE;
/* Here we figure out which node 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(gtkblist->treemodel), &iter, path);
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
/* Right click draws a context menu */
if ((event->button == 3) && (event->type == GDK_BUTTON_PRESS)) {
handled = gaim_gtk_blist_show_context_menu(node, NULL, tv, 3, event->time);
/* CTRL+middle click expands or collapse a contact */
} else if ((event->button == 2) && (event->type == GDK_BUTTON_PRESS) &&
(event->state & GDK_CONTROL_MASK) && (GAIM_BLIST_NODE_IS_CONTACT(node))) {
if (gtknode->contact_expanded)
gaim_gtk_blist_collapse_contact_cb(NULL, node);
else
gaim_gtk_blist_expand_contact_cb(NULL, node);
handled = TRUE;
/* Double middle click gets info */
} else if ((event->button == 2) && (event->type == GDK_2BUTTON_PRESS) &&
((GAIM_BLIST_NODE_IS_CONTACT(node)) || (GAIM_BLIST_NODE_IS_BUDDY(node)))) {
GaimBuddy *b;
if(GAIM_BLIST_NODE_IS_CONTACT(node))
b = gaim_contact_get_priority_buddy((GaimContact*)node);
else
b = (GaimBuddy *)node;
prpl = gaim_find_prpl(gaim_account_get_protocol_id(b->account));
if (prpl != NULL)
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if (prpl && prpl_info->get_info)
serv_get_info(b->account->gc, b->name);
handled = TRUE;
}
#if (1)
/*
* This code only exists because GTK+ doesn't work. If we return
* FALSE here, as would be normal the event propoagates down and
* somehow gets interpreted as the start of a drag event.
*
* Um, isn't it _normal_ to return TRUE here? Since the event
* was handled? --Mark
*/
if(handled) {
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv));
gtk_tree_selection_select_path(sel, path);
gtk_tree_path_free(path);
return TRUE;
}
#endif
gtk_tree_path_free(path);
return FALSE;
}
static gboolean
gaim_gtk_blist_popup_menu_cb(GtkWidget *tv, void *user_data)
{
GaimBlistNode *node;
GValue val;
GtkTreeIter iter;
GtkTreeSelection *sel;
gboolean handled = FALSE;
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv));
if (!gtk_tree_selection_get_selected(sel, NULL, &iter))
return FALSE;
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel),
&iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
/* Shift+F10 draws a context menu */
handled = gaim_gtk_blist_show_context_menu(node, gaim_gtk_treeview_popup_menu_position_func, tv, 0, GDK_CURRENT_TIME);
return handled;
}
static void gaim_gtk_blist_buddy_details_cb(gpointer data, guint action, GtkWidget *item)
{
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
gaim_prefs_set_bool("/gaim/gtk/blist/show_buddy_icons",
gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(item)));
gaim_gtk_clear_cursor(gtkblist->window);
}
static void gaim_gtk_blist_show_idle_time_cb(gpointer data, guint action, GtkWidget *item)
{
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
gaim_prefs_set_bool("/gaim/gtk/blist/show_idle_time",
gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(item)));
gaim_gtk_clear_cursor(gtkblist->window);
}
static void gaim_gtk_blist_show_empty_groups_cb(gpointer data, guint action, GtkWidget *item)
{
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
gaim_prefs_set_bool("/gaim/gtk/blist/show_empty_groups",
gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(item)));
gaim_gtk_clear_cursor(gtkblist->window);
}
static void gaim_gtk_blist_edit_mode_cb(gpointer callback_data, guint callback_action,
GtkWidget *checkitem)
{
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
gaim_prefs_set_bool("/gaim/gtk/blist/show_offline_buddies",
gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(checkitem)));
gaim_gtk_clear_cursor(gtkblist->window);
}
static void gaim_gtk_blist_mute_sounds_cb(gpointer data, guint action, GtkWidget *item)
{
gaim_prefs_set_bool("/gaim/gtk/sound/mute", GTK_CHECK_MENU_ITEM(item)->active);
}
static void
gaim_gtk_blist_mute_pref_cb(const char *name, GaimPrefType type,
gconstpointer value, gpointer data)
{
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item(gtkblist->ift,
N_("/Tools/Mute Sounds"))), (gboolean)GPOINTER_TO_INT(value));
}
static void
gaim_gtk_blist_sound_method_pref_cb(const char *name, GaimPrefType type,
gconstpointer value, gpointer data)
{
gboolean sensitive = TRUE;
if(!strcmp(value, "none"))
sensitive = FALSE;
gtk_widget_set_sensitive(gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools/Mute Sounds")), sensitive);
}
static void
add_buddies_from_vcard(const char *prpl_id, GaimGroup *group, GList *list,
const char *alias)
{
GList *l;
GaimAccount *account = NULL;
GaimConnection *gc;
if (list == NULL)
return;
for (l = gaim_connections_get_all(); l != NULL; l = l->next)
{
gc = (GaimConnection *)l->data;
account = gaim_connection_get_account(gc);
if (!strcmp(gaim_account_get_protocol_id(account), prpl_id))
break;
account = NULL;
}
if (account != NULL)
{
for (l = list; l != NULL; l = l->next)
{
gaim_blist_request_add_buddy(account, l->data,
(group ? group->name : NULL),
alias);
}
}
g_list_foreach(list, (GFunc)g_free, NULL);
g_list_free(list);
}
static gboolean
parse_vcard(const char *vcard, GaimGroup *group)
{
char *temp_vcard;
char *s, *c;
char *alias = NULL;
GList *aims = NULL;
GList *icqs = NULL;
GList *yahoos = NULL;
GList *msns = NULL;
GList *jabbers = NULL;
s = temp_vcard = g_strdup(vcard);
while (*s != '\0' && strncmp(s, "END:vCard", strlen("END:vCard")))
{
char *field, *value;
field = s;
/* Grab the field */
while (*s != '\r' && *s != '\n' && *s != '\0' && *s != ':')
s++;
if (*s == '\r') s++;
if (*s == '\n')
{
s++;
continue;
}
if (*s != '\0') *s++ = '\0';
if ((c = strchr(field, ';')) != NULL)
*c = '\0';
/* Proceed to the end of the line */
value = s;
while (*s != '\r' && *s != '\n' && *s != '\0')
s++;
if (*s == '\r') *s++ = '\0';
if (*s == '\n') *s++ = '\0';
/* We only want to worry about a few fields here. */
if (!strcmp(field, "FN"))
alias = g_strdup(value);
else if (!strcmp(field, "X-AIM") || !strcmp(field, "X-ICQ") ||
!strcmp(field, "X-YAHOO") || !strcmp(field, "X-MSN") ||
!strcmp(field, "X-JABBER"))
{
char **values = g_strsplit(value, ":", 0);
char **im;
for (im = values; *im != NULL; im++)
{
if (!strcmp(field, "X-AIM"))
aims = g_list_append(aims, g_strdup(*im));
else if (!strcmp(field, "X-ICQ"))
icqs = g_list_append(icqs, g_strdup(*im));
else if (!strcmp(field, "X-YAHOO"))
yahoos = g_list_append(yahoos, g_strdup(*im));
else if (!strcmp(field, "X-MSN"))
msns = g_list_append(msns, g_strdup(*im));
else if (!strcmp(field, "X-JABBER"))
jabbers = g_list_append(jabbers, g_strdup(*im));
}
g_strfreev(values);
}
}
g_free(temp_vcard);
if (aims == NULL && icqs == NULL && yahoos == NULL &&
msns == NULL && jabbers == NULL)
{
g_free(alias);
return FALSE;
}
add_buddies_from_vcard("prpl-oscar", group, aims, alias);
add_buddies_from_vcard("prpl-oscar", group, icqs, alias);
add_buddies_from_vcard("prpl-yahoo", group, yahoos, alias);
add_buddies_from_vcard("prpl-msn", group, msns, alias);
add_buddies_from_vcard("prpl-jabber", group, jabbers, alias);
g_free(alias);
return TRUE;
}
#ifdef _WIN32
static void gaim_gtk_blist_drag_begin(GtkWidget *widget,
GdkDragContext *drag_context, gpointer user_data)
{
gaim_gtk_blist_tooltip_destroy();
/* Unhook the tooltip-timeout since we don't want a tooltip
* to appear and obscure the dragging operation.
* This is a workaround for GTK+ bug 107320. */
if (gtkblist->timeout) {
g_source_remove(gtkblist->timeout);
gtkblist->timeout = 0;
}
}
#endif
static void gaim_gtk_blist_drag_data_get_cb(GtkWidget *widget,
GdkDragContext *dc,
GtkSelectionData *data,
guint info,
guint time,
gpointer null)
{
if (data->target == gdk_atom_intern("GAIM_BLIST_NODE", FALSE))
{
GtkTreeRowReference *ref = g_object_get_data(G_OBJECT(dc), "gtk-tree-view-source-row");
GtkTreePath *sourcerow = gtk_tree_row_reference_get_path(ref);
GtkTreeIter iter;
GaimBlistNode *node = NULL;
GValue val;
if(!sourcerow)
return;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &iter, sourcerow);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
gtk_selection_data_set (data,
gdk_atom_intern ("GAIM_BLIST_NODE", FALSE),
8, /* bits */
(void*)&node,
sizeof (node));
gtk_tree_path_free(sourcerow);
}
else if (data->target == gdk_atom_intern("application/x-im-contact", FALSE))
{
GtkTreeRowReference *ref;
GtkTreePath *sourcerow;
GtkTreeIter iter;
GaimBlistNode *node = NULL;
GaimBuddy *buddy;
GaimConnection *gc;
GValue val;
GString *str;
const char *protocol;
ref = g_object_get_data(G_OBJECT(dc), "gtk-tree-view-source-row");
sourcerow = gtk_tree_row_reference_get_path(ref);
if (!sourcerow)
return;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &iter,
sourcerow);
val.g_type = 0;
gtk_tree_model_get_value(GTK_TREE_MODEL(gtkblist->treemodel), &iter,
NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_CONTACT(node))
{
buddy = gaim_contact_get_priority_buddy((GaimContact *)node);
}
else if (!GAIM_BLIST_NODE_IS_BUDDY(node))
{
gtk_tree_path_free(sourcerow);
return;
}
else
{
buddy = (GaimBuddy *)node;
}
gc = gaim_account_get_connection(buddy->account);
if (gc == NULL)
{
gtk_tree_path_free(sourcerow);
return;
}
protocol =
GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->list_icon(buddy->account,
buddy);
str = g_string_new(NULL);
g_string_printf(str,
"MIME-Version: 1.0\r\n"
"Content-Type: application/x-im-contact\r\n"
"X-IM-Protocol: %s\r\n"
"X-IM-Username: %s\r\n",
protocol,
buddy->name);
if (buddy->alias != NULL)
{
g_string_append_printf(str,
"X-IM-Alias: %s\r\n",
buddy->alias);
}
g_string_append(str, "\r\n");
gtk_selection_data_set(data,
gdk_atom_intern("application/x-im-contact", FALSE),
8, /* bits */
(const guchar *)str->str,
strlen(str->str) + 1);
g_string_free(str, TRUE);
gtk_tree_path_free(sourcerow);
}
}
static void gaim_gtk_blist_drag_data_rcv_cb(GtkWidget *widget, GdkDragContext *dc, guint x, guint y,
GtkSelectionData *sd, guint info, guint t)
{
if (gtkblist->drag_timeout) {
g_source_remove(gtkblist->drag_timeout);
gtkblist->drag_timeout = 0;
}
if (sd->target == gdk_atom_intern("GAIM_BLIST_NODE", FALSE) && sd->data) {
GaimBlistNode *n = NULL;
GtkTreePath *path = NULL;
GtkTreeViewDropPosition position;
memcpy(&n, sd->data, sizeof(n));
if(gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(widget), x, y, &path, &position)) {
/* if we're here, I think it means the drop is ok */
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
struct _gaim_gtk_blist_node *gtknode;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel),
&iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel),
&iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
gtknode = node->ui_data;
if (GAIM_BLIST_NODE_IS_CONTACT(n)) {
GaimContact *c = (GaimContact*)n;
if (GAIM_BLIST_NODE_IS_CONTACT(node) && gtknode->contact_expanded) {
gaim_blist_merge_contact(c, node);
} else if (GAIM_BLIST_NODE_IS_CONTACT(node) ||
GAIM_BLIST_NODE_IS_CHAT(node)) {
switch(position) {
case GTK_TREE_VIEW_DROP_AFTER:
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
gaim_blist_add_contact(c, (GaimGroup*)node->parent,
node);
break;
case GTK_TREE_VIEW_DROP_BEFORE:
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
gaim_blist_add_contact(c, (GaimGroup*)node->parent,
node->prev);
break;
}
} else if(GAIM_BLIST_NODE_IS_GROUP(node)) {
gaim_blist_add_contact(c, (GaimGroup*)node, NULL);
} else if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
gaim_blist_merge_contact(c, node);
}
} else if (GAIM_BLIST_NODE_IS_BUDDY(n)) {
GaimBuddy *b = (GaimBuddy*)n;
if (GAIM_BLIST_NODE_IS_BUDDY(node)) {
switch(position) {
case GTK_TREE_VIEW_DROP_AFTER:
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
gaim_blist_add_buddy(b, (GaimContact*)node->parent,
(GaimGroup*)node->parent->parent, node);
break;
case GTK_TREE_VIEW_DROP_BEFORE:
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
gaim_blist_add_buddy(b, (GaimContact*)node->parent,
(GaimGroup*)node->parent->parent,
node->prev);
break;
}
} else if(GAIM_BLIST_NODE_IS_CHAT(node)) {
gaim_blist_add_buddy(b, NULL, (GaimGroup*)node->parent,
NULL);
} else if (GAIM_BLIST_NODE_IS_GROUP(node)) {
gaim_blist_add_buddy(b, NULL, (GaimGroup*)node, NULL);
} else if (GAIM_BLIST_NODE_IS_CONTACT(node)) {
if(gtknode->contact_expanded) {
switch(position) {
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
case GTK_TREE_VIEW_DROP_AFTER:
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
gaim_blist_add_buddy(b, (GaimContact*)node,
(GaimGroup*)node->parent, NULL);
break;
case GTK_TREE_VIEW_DROP_BEFORE:
gaim_blist_add_buddy(b, NULL,
(GaimGroup*)node->parent, node->prev);
break;
}
} else {
switch(position) {
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
case GTK_TREE_VIEW_DROP_AFTER:
gaim_blist_add_buddy(b, NULL,
(GaimGroup*)node->parent, NULL);
break;
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
case GTK_TREE_VIEW_DROP_BEFORE:
gaim_blist_add_buddy(b, NULL,
(GaimGroup*)node->parent, node->prev);
break;
}
}
}
} else if (GAIM_BLIST_NODE_IS_CHAT(n)) {
GaimChat *chat = (GaimChat *)n;
if (GAIM_BLIST_NODE_IS_BUDDY(node)) {
switch(position) {
case GTK_TREE_VIEW_DROP_AFTER:
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
case GTK_TREE_VIEW_DROP_BEFORE:
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
gaim_blist_add_chat(chat,
(GaimGroup*)node->parent->parent,
node->parent);
break;
}
} else if(GAIM_BLIST_NODE_IS_CONTACT(node) ||
GAIM_BLIST_NODE_IS_CHAT(node)) {
switch(position) {
case GTK_TREE_VIEW_DROP_AFTER:
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
gaim_blist_add_chat(chat, (GaimGroup*)node->parent, node);
break;
case GTK_TREE_VIEW_DROP_BEFORE:
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
gaim_blist_add_chat(chat, (GaimGroup*)node->parent, node->prev);
break;
}
} else if (GAIM_BLIST_NODE_IS_GROUP(node)) {
gaim_blist_add_chat(chat, (GaimGroup*)node, NULL);
}
} else if (GAIM_BLIST_NODE_IS_GROUP(n)) {
GaimGroup *g = (GaimGroup*)n;
if (GAIM_BLIST_NODE_IS_GROUP(node)) {
switch (position) {
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
case GTK_TREE_VIEW_DROP_AFTER:
gaim_blist_add_group(g, node);
break;
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
case GTK_TREE_VIEW_DROP_BEFORE:
gaim_blist_add_group(g, node->prev);
break;
}
} else if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
gaim_blist_add_group(g, node->parent->parent);
} else if(GAIM_BLIST_NODE_IS_CONTACT(node) ||
GAIM_BLIST_NODE_IS_CHAT(node)) {
gaim_blist_add_group(g, node->parent);
}
}
gtk_tree_path_free(path);
gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
}
}
else if (sd->target == gdk_atom_intern("application/x-im-contact",
FALSE) && sd->data)
{
GaimGroup *group = NULL;
GtkTreePath *path = NULL;
GtkTreeViewDropPosition position;
GaimAccount *account;
char *protocol = NULL;
char *username = NULL;
char *alias = NULL;
if (gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(widget),
x, y, &path, &position))
{
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel),
&iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel),
&iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_BUDDY(node))
{
group = (GaimGroup *)node->parent->parent;
}
else if (GAIM_BLIST_NODE_IS_CHAT(node) ||
GAIM_BLIST_NODE_IS_CONTACT(node))
{
group = (GaimGroup *)node->parent;
}
else if (GAIM_BLIST_NODE_IS_GROUP(node))
{
group = (GaimGroup *)node;
}
}
if (gaim_gtk_parse_x_im_contact((const char *)sd->data, FALSE, &account,
&protocol, &username, &alias))
{
if (account == NULL)
{
gaim_notify_error(NULL, NULL,
_("You are not currently signed on with an account that "
"can add that buddy."), NULL);
}
else
{
gaim_blist_request_add_buddy(account, username,
(group ? group->name : NULL),
alias);
}
}
g_free(username);
g_free(protocol);
g_free(alias);
if (path != NULL)
gtk_tree_path_free(path);
gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
}
else if (sd->target == gdk_atom_intern("text/x-vcard", FALSE) && sd->data)
{
gboolean result;
GaimGroup *group = NULL;
GtkTreePath *path = NULL;
GtkTreeViewDropPosition position;
if (gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(widget),
x, y, &path, &position))
{
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel),
&iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel),
&iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_BUDDY(node))
{
group = (GaimGroup *)node->parent->parent;
}
else if (GAIM_BLIST_NODE_IS_CHAT(node) ||
GAIM_BLIST_NODE_IS_CONTACT(node))
{
group = (GaimGroup *)node->parent;
}
else if (GAIM_BLIST_NODE_IS_GROUP(node))
{
group = (GaimGroup *)node;
}
}
result = parse_vcard((const gchar *)sd->data, group);
gtk_drag_finish(dc, result, (dc->action == GDK_ACTION_MOVE), t);
} else if (sd->target == gdk_atom_intern("text/uri-list", FALSE) && sd->data) {
GtkTreePath *path = NULL;
GtkTreeViewDropPosition position;
if (gtk_tree_view_get_dest_row_at_pos(GTK_TREE_VIEW(widget),
x, y, &path, &position))
{
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel),
&iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel),
&iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if (GAIM_BLIST_NODE_IS_BUDDY(node) || GAIM_BLIST_NODE_IS_CONTACT(node)) {
GaimBuddy *b = GAIM_BLIST_NODE_IS_BUDDY(node) ? (GaimBuddy*)node : gaim_contact_get_priority_buddy((GaimContact*)node);
gaim_dnd_file_manage(sd, b->account, b->name);
gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
} else {
gtk_drag_finish(dc, FALSE, FALSE, t);
}
}
}
}
static GdkPixbuf *gaim_gtk_blist_get_buddy_icon(GaimBlistNode *node,
gboolean scaled, gboolean greyed, gboolean custom)
{
GdkPixbuf *buf, *ret = NULL;
GdkPixbufLoader *loader;
GaimBuddyIcon *icon;
const guchar *data = NULL;
gsize len;
GaimBuddy *buddy = (GaimBuddy *)node;
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
buddy = gaim_contact_get_priority_buddy((GaimContact*)node);
} else if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
buddy = (GaimBuddy*)node;
} else {
return NULL;
}
#if 0
if (!gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"))
return NULL;
#endif
if (custom) {
const char *file = gaim_blist_node_get_string((GaimBlistNode*)gaim_buddy_get_contact(buddy),
"custom_buddy_icon");
if (file && *file) {
char *contents;
GError *err = NULL;
if (!g_file_get_contents(file, &contents, &len, &err)) {
gaim_debug_info("custom -icon", "Could not open custom-icon %s for %s\n",
file, gaim_buddy_get_name(buddy), err->message);
g_error_free(err);
} else
data = (const guchar*)contents;
}
}
if (data == NULL) {
if (!(icon = gaim_buddy_get_icon(buddy)))
if (!(icon = gaim_buddy_icons_find(buddy->account, buddy->name))) /* Not sure I like this...*/
return NULL;
data = gaim_buddy_icon_get_data(icon, &len);
custom = FALSE; /* We are not using the custom icon */
}
loader = gdk_pixbuf_loader_new();
gdk_pixbuf_loader_write(loader, data, len, NULL);
gdk_pixbuf_loader_close(loader, NULL);
buf = gdk_pixbuf_loader_get_pixbuf(loader);
if (buf)
g_object_ref(G_OBJECT(buf));
g_object_unref(G_OBJECT(loader));
if (custom)
g_free((void*)data);
if (buf) {
GaimAccount *account = gaim_buddy_get_account(buddy);
GaimPluginProtocolInfo *prpl_info = NULL;
int orig_width, orig_height;
int scale_width, scale_height;
if(account && account->gc)
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(account->gc->prpl);
if (greyed) {
GaimPresence *presence = gaim_buddy_get_presence(buddy);
if (!GAIM_BUDDY_IS_ONLINE(buddy))
gdk_pixbuf_saturate_and_pixelate(buf, buf, 0.0, FALSE);
if (gaim_presence_is_idle(presence))
gdk_pixbuf_saturate_and_pixelate(buf, buf, 0.25, FALSE);
}
/* i'd use the gaim_gtk_buddy_icon_get_scale_size() thing,
* but it won't tell me the original size, which I need for scaling
* purposes */
scale_width = orig_width = gdk_pixbuf_get_width(buf);
scale_height = orig_height = gdk_pixbuf_get_height(buf);
if (prpl_info && prpl_info->icon_spec.scale_rules & GAIM_ICON_SCALE_DISPLAY)
gaim_buddy_icon_get_scale_size(&prpl_info->icon_spec, &scale_width, &scale_height);
if (scaled) {
if(scale_height > scale_width) {
scale_width = 30.0 * (double)scale_width / (double)scale_height;
scale_height = 30;
} else {
scale_height = 30.0 * (double)scale_height / (double)scale_width;
scale_width = 30;
}
ret = gdk_pixbuf_new(GDK_COLORSPACE_RGB, TRUE, 8, 30, 30);
gdk_pixbuf_fill(ret, 0x00000000);
gdk_pixbuf_scale(buf, ret, (30-scale_width)/2, (30-scale_height)/2, scale_width, scale_height, (30-scale_width)/2, (30-scale_height)/2, (double)scale_width/(double)orig_width, (double)scale_height/(double)orig_height, GDK_INTERP_BILINEAR);
} else {
ret = gdk_pixbuf_scale_simple(buf,scale_width,scale_height, GDK_INTERP_BILINEAR);
}
g_object_unref(G_OBJECT(buf));
}
return ret;
}
struct tooltip_data {
PangoLayout *layout;
GdkPixbuf *status_icon;
GdkPixbuf *avatar;
int avatar_width;
int width;
int height;
};
static struct tooltip_data * create_tip_for_node(GaimBlistNode *node, gboolean full)
{
char *tooltip_text = NULL;
struct tooltip_data *td = g_new0(struct tooltip_data, 1);
td->status_icon = gaim_gtk_blist_get_status_icon(node, GAIM_STATUS_ICON_LARGE);
td->avatar = gaim_gtk_blist_get_buddy_icon(node, !full, FALSE, FALSE);
tooltip_text = gaim_get_tooltip_text(node, full);
td->layout = gtk_widget_create_pango_layout(gtkblist->tipwindow, NULL);
pango_layout_set_markup(td->layout, tooltip_text, -1);
pango_layout_set_wrap(td->layout, PANGO_WRAP_WORD);
pango_layout_set_width(td->layout, 300000);
pango_layout_get_size (td->layout, &td->width, &td->height);
td->width = PANGO_PIXELS(td->width) + 38 + 8;
td->height = MAX(PANGO_PIXELS(td->height + 4) + 8, 38);
if(td->avatar) {
td->avatar_width = gdk_pixbuf_get_width(td->avatar);
td->width += td->avatar_width + 8;
td->height = MAX(td->height, gdk_pixbuf_get_height(td->avatar) + 8);
}
g_free(tooltip_text);
return td;
}
static void gaim_gtk_blist_paint_tip(GtkWidget *widget, GdkEventExpose *event, GaimBlistNode *node)
{
GtkStyle *style;
int current_height, max_width;
GList *l;
if(gtkblist->tooltipdata == NULL)
return;
style = gtkblist->tipwindow->style;
gtk_paint_flat_box(style, gtkblist->tipwindow->window, GTK_STATE_NORMAL, GTK_SHADOW_OUT,
NULL, gtkblist->tipwindow, "tooltip", 0, 0, -1, -1);
max_width = 0;
for(l = gtkblist->tooltipdata; l; l = l->next)
{
struct tooltip_data *td = l->data;
max_width = MAX(max_width, td->width);
}
current_height = 4;
for(l = gtkblist->tooltipdata; l; l = l->next)
{
struct tooltip_data *td = l->data;
#if GTK_CHECK_VERSION(2,2,0)
gdk_draw_pixbuf(GDK_DRAWABLE(gtkblist->tipwindow->window), NULL, td->status_icon,
0, 0, 4, current_height, -1 , -1, GDK_RGB_DITHER_NONE, 0, 0);
if(td->avatar)
gdk_draw_pixbuf(GDK_DRAWABLE(gtkblist->tipwindow->window), NULL,
td->avatar, 0, 0, max_width - (td->avatar_width + 4), current_height, -1 , -1, GDK_RGB_DITHER_NONE, 0, 0);
#else
gdk_pixbuf_render_to_drawable(td->status_icon, GDK_DRAWABLE(gtkblist->tipwindow->window), NULL, 0, 0, 4, current_height, -1, -1, GDK_RGB_DITHER_NONE, 0, 0);
if(td->avatar)
gdk_pixbuf_render_to_drawable(td->avatar,
GDK_DRAWABLE(gtkblist->tipwindow->window), NULL, 0, 0,
max_width - (td->avatar_width + 4),
current_height, -1, -1, GDK_RGB_DITHER_NONE, 0, 0);
#endif
gtk_paint_layout (style, gtkblist->tipwindow->window, GTK_STATE_NORMAL, FALSE,
NULL, gtkblist->tipwindow, "tooltip", 38 + 4, current_height, td->layout);
current_height += td->height;
if(l->next)
gtk_paint_hline(style, gtkblist->tipwindow->window, GTK_STATE_NORMAL, NULL, NULL, NULL, 4, max_width - 4, current_height-6);
}
}
static void gaim_gtk_blist_tooltip_destroy()
{
while(gtkblist->tooltipdata) {
struct tooltip_data *td = gtkblist->tooltipdata->data;
if(td->avatar)
g_object_unref(td->avatar);
if(td->status_icon)
g_object_unref(td->status_icon);
g_object_unref(td->layout);
g_free(td);
gtkblist->tooltipdata = g_list_delete_link(gtkblist->tooltipdata, gtkblist->tooltipdata);
}
if (gtkblist->tipwindow == NULL)
return;
gtk_widget_destroy(gtkblist->tipwindow);
gtkblist->tipwindow = NULL;
}
static gboolean gaim_gtk_blist_expand_timeout(GtkWidget *tv)
{
GtkTreePath *path;
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
struct _gaim_gtk_blist_node *gtknode;
if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), gtkblist->tip_rect.x, gtkblist->tip_rect.y, &path, NULL, NULL, NULL))
return FALSE;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
if(!GAIM_BLIST_NODE_IS_CONTACT(node)) {
gtk_tree_path_free(path);
return FALSE;
}
gtknode = node->ui_data;
if (!gtknode->contact_expanded) {
GtkTreeIter i;
gaim_gtk_blist_expand_contact_cb(NULL, node);
gtk_tree_view_get_cell_area(GTK_TREE_VIEW(tv), path, NULL, &gtkblist->contact_rect);
gdk_drawable_get_size(GDK_DRAWABLE(tv->window), &(gtkblist->contact_rect.width), NULL);
gtkblist->mouseover_contact = node;
gtk_tree_path_down (path);
while (gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &i, path)) {
GdkRectangle rect;
gtk_tree_view_get_cell_area(GTK_TREE_VIEW(tv), path, NULL, &rect);
gtkblist->contact_rect.height += rect.height;
gtk_tree_path_next(path);
}
}
gtk_tree_path_free(path);
return FALSE;
}
static gboolean buddy_is_displayable(GaimBuddy *buddy)
{
struct _gaim_gtk_blist_node *gtknode;
if(!buddy)
return FALSE;
gtknode = ((GaimBlistNode*)buddy)->ui_data;
return (gaim_account_is_connected(buddy->account) &&
(gaim_presence_is_online(buddy->presence) ||
(gtknode && gtknode->recent_signonoff) ||
gaim_prefs_get_bool("/gaim/gtk/blist/show_offline_buddies") ||
gaim_blist_node_get_bool((GaimBlistNode*)buddy, "show_offline")));
}
static gboolean gaim_gtk_blist_tooltip_timeout(GtkWidget *tv)
{
GtkTreePath *path;
GtkTreeIter iter;
GaimBlistNode *node;
GValue val;
int scr_w, scr_h, w, h, x, y;
#if GTK_CHECK_VERSION(2,2,0)
int mon_num;
GdkScreen *screen = NULL;
#endif
gboolean tooltip_top = FALSE;
struct _gaim_gtk_blist_node *gtknode;
GdkRectangle mon_size;
/*
* Attempt to free the previous tooltip. I have a feeling
* this is never needed... but just in case.
*/
gaim_gtk_blist_tooltip_destroy();
if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), gtkblist->tip_rect.x, gtkblist->tip_rect.y, &path, NULL, NULL, NULL))
return FALSE;
gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), &iter, path);
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &val);
node = g_value_get_pointer(&val);
gtk_tree_path_free(path);
gtkblist->tipwindow = gtk_window_new(GTK_WINDOW_POPUP);
if(GAIM_BLIST_NODE_IS_CHAT(node) || GAIM_BLIST_NODE_IS_BUDDY(node)) {
struct tooltip_data *td = create_tip_for_node(node, TRUE);
gtkblist->tooltipdata = g_list_append(gtkblist->tooltipdata, td);
w = td->width;
h = td->height;
} else if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
GaimBlistNode *child;
GaimBuddy *b = gaim_contact_get_priority_buddy((GaimContact *)node);
w = h = 0;
for(child = node->child; child; child = child->next)
{
if(GAIM_BLIST_NODE_IS_BUDDY(child) && buddy_is_displayable((GaimBuddy*)child)) {
struct tooltip_data *td = create_tip_for_node(child, (b == (GaimBuddy*)child));
if (b == (GaimBuddy *)child) {
gtkblist->tooltipdata = g_list_prepend(gtkblist->tooltipdata, td);
} else {
gtkblist->tooltipdata = g_list_append(gtkblist->tooltipdata, td);
}
w = MAX(w, td->width);
h += td->height;
}
}
} else {
gtk_widget_destroy(gtkblist->tipwindow);
gtkblist->tipwindow = NULL;
return FALSE;
}
if (gtkblist->tooltipdata == NULL) {
gtk_widget_destroy(gtkblist->tipwindow);
gtkblist->tipwindow = NULL;
return FALSE;
}
gtknode = node->ui_data;
gtk_widget_set_app_paintable(gtkblist->tipwindow, TRUE);
gtk_window_set_resizable(GTK_WINDOW(gtkblist->tipwindow), FALSE);
gtk_widget_set_name(gtkblist->tipwindow, "gtk-tooltips");
g_signal_connect(G_OBJECT(gtkblist->tipwindow), "expose_event",
G_CALLBACK(gaim_gtk_blist_paint_tip), NULL);
gtk_widget_ensure_style (gtkblist->tipwindow);
#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
if (GTK_WIDGET_NO_WINDOW(gtkblist->window))
y+=gtkblist->window->allocation.y;
x -= ((w >> 1) + 4);
if ((y + h + 4) > scr_h || tooltip_top)
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(gtkblist->tipwindow, w, h);
gtk_window_move(GTK_WINDOW(gtkblist->tipwindow), x, y);
gtk_widget_show(gtkblist->tipwindow);
return FALSE;
}
static gboolean gaim_gtk_blist_drag_motion_cb(GtkWidget *tv, GdkDragContext *drag_context,
gint x, gint y, guint time, gpointer user_data)
{
GtkTreePath *path;
int delay;
/*
* When dragging a buddy into a contact, this is the delay before
* the contact auto-expands.
*/
delay = 900;
if (gtkblist->drag_timeout) {
if ((y > gtkblist->tip_rect.y) && ((y - gtkblist->tip_rect.height) < gtkblist->tip_rect.y))
return FALSE;
/* We've left the cell. Remove the timeout and create a new one below */
g_source_remove(gtkblist->drag_timeout);
}
gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), x, y, &path, NULL, NULL, NULL);
gtk_tree_view_get_cell_area(GTK_TREE_VIEW(tv), path, NULL, &gtkblist->tip_rect);
if (path)
gtk_tree_path_free(path);
gtkblist->drag_timeout = g_timeout_add(delay, (GSourceFunc)gaim_gtk_blist_expand_timeout, tv);
if (gtkblist->mouseover_contact) {
if ((y < gtkblist->contact_rect.y) || ((y - gtkblist->contact_rect.height) > gtkblist->contact_rect.y)) {
gaim_gtk_blist_collapse_contact_cb(NULL, gtkblist->mouseover_contact);
gtkblist->mouseover_contact = NULL;
}
}
return FALSE;
}
static gboolean gaim_gtk_blist_motion_cb (GtkWidget *tv, GdkEventMotion *event, gpointer null)
{
GtkTreePath *path;
int delay;
delay = gaim_prefs_get_int("/gaim/gtk/blist/tooltip_delay");
if (delay == 0)
return FALSE;
if (gtkblist->timeout) {
if ((event->y > gtkblist->tip_rect.y) && ((event->y - gtkblist->tip_rect.height) < gtkblist->tip_rect.y))
return FALSE;
/* We've left the cell. Remove the timeout and create a new one below */
gaim_gtk_blist_tooltip_destroy();
g_source_remove(gtkblist->timeout);
}
gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(tv), event->x, event->y, &path, NULL, NULL, NULL);
gtk_tree_view_get_cell_area(GTK_TREE_VIEW(tv), path, NULL, &gtkblist->tip_rect);
if (path)
gtk_tree_path_free(path);
gtkblist->timeout = g_timeout_add(delay, (GSourceFunc)gaim_gtk_blist_tooltip_timeout, tv);
if (gtkblist->mouseover_contact) {
if ((event->y < gtkblist->contact_rect.y) || ((event->y - gtkblist->contact_rect.height) > gtkblist->contact_rect.y)) {
gaim_gtk_blist_collapse_contact_cb(NULL, gtkblist->mouseover_contact);
gtkblist->mouseover_contact = NULL;
}
}
return FALSE;
}
static void gaim_gtk_blist_leave_cb (GtkWidget *w, GdkEventCrossing *e, gpointer n)
{
if (gtkblist->timeout) {
g_source_remove(gtkblist->timeout);
gtkblist->timeout = 0;
}
if (gtkblist->drag_timeout) {
g_source_remove(gtkblist->drag_timeout);
gtkblist->drag_timeout = 0;
}
gaim_gtk_blist_tooltip_destroy();
if (gtkblist->mouseover_contact &&
!((e->x > gtkblist->contact_rect.x) && (e->x < (gtkblist->contact_rect.x + gtkblist->contact_rect.width)) &&
(e->y > gtkblist->contact_rect.y) && (e->y < (gtkblist->contact_rect.y + gtkblist->contact_rect.height)))) {
gaim_gtk_blist_collapse_contact_cb(NULL, gtkblist->mouseover_contact);
gtkblist->mouseover_contact = NULL;
}
}
static void
toggle_debug(void)
{
gaim_prefs_set_bool("/gaim/gtk/debug/enabled",
!gaim_prefs_get_bool("/gaim/gtk/debug/enabled"));
}
/***************************************************
* Crap *
***************************************************/
static GtkItemFactoryEntry blist_menu[] =
{
/* Buddies menu */
{ N_("/_Buddies"), NULL, NULL, 0, "<Branch>", NULL },
{ N_("/Buddies/New Instant _Message..."), "<CTL>M", gaim_gtkdialogs_im, 0, "<StockItem>", GAIM_STOCK_IM },
{ N_("/Buddies/Join a _Chat..."), "<CTL>C", gaim_gtk_blist_joinchat_show, 0, "<StockItem>", GAIM_STOCK_CHAT },
{ N_("/Buddies/Get User _Info..."), "<CTL>I", gaim_gtkdialogs_info, 0, "<StockItem>", GAIM_STOCK_INFO },
{ N_("/Buddies/View User _Log..."), "<CTL>L", gaim_gtkdialogs_log, 0, "<StockItem>", GAIM_STOCK_LOG },
{ "/Buddies/sep1", NULL, NULL, 0, "<Separator>", NULL },
{ N_("/Buddies/Show _Offline Buddies"), NULL, gaim_gtk_blist_edit_mode_cb, 1, "<CheckItem>", NULL },
{ N_("/Buddies/Show _Empty Groups"), NULL, gaim_gtk_blist_show_empty_groups_cb, 1, "<CheckItem>", NULL },
{ N_("/Buddies/Show Buddy _Details"), NULL, gaim_gtk_blist_buddy_details_cb, 1, "<CheckItem>", NULL },
{ N_("/Buddies/Show Idle _Times"), NULL, gaim_gtk_blist_show_idle_time_cb, 1, "<CheckItem>", NULL },
{ N_("/Buddies/_Sort Buddies"), NULL, NULL, 0, "<Branch>", NULL },
{ "/Buddies/sep2", NULL, NULL, 0, "<Separator>", NULL },
{ N_("/Buddies/_Add Buddy..."), "<CTL>B", gaim_gtk_blist_add_buddy_cb, 0, "<StockItem>", GTK_STOCK_ADD },
{ N_("/Buddies/Add C_hat..."), NULL, gaim_gtk_blist_add_chat_cb, 0, "<StockItem>", GTK_STOCK_ADD },
{ N_("/Buddies/Add _Group..."), NULL, gaim_blist_request_add_group, 0, "<StockItem>", GTK_STOCK_ADD },
{ "/Buddies/sep3", NULL, NULL, 0, "<Separator>", NULL },
{ N_("/Buddies/_Quit"), "<CTL>Q", gaim_core_quit, 0, "<StockItem>", GTK_STOCK_QUIT },
/* Accounts menu */
{ N_("/_Accounts"), NULL, NULL, 0, "<Branch>", NULL },
{ N_("/Accounts/Add\\/Edit"), "<CTL>A", gaim_gtk_accounts_window_show, 0, "<StockItem>", GAIM_STOCK_ACCOUNTS },
/* Tools */
{ N_("/_Tools"), NULL, NULL, 0, "<Branch>", NULL },
{ N_("/Tools/Buddy _Pounces"), NULL, gaim_gtk_pounces_manager_show, 0, "<StockItem>", GAIM_STOCK_POUNCE },
{ N_("/Tools/Plu_gins"), "<CTL>U", gaim_gtk_plugin_dialog_show, 0, "<StockItem>", GAIM_STOCK_PLUGIN },
{ N_("/Tools/Pr_eferences"), "<CTL>P", gaim_gtk_prefs_show, 0, "<StockItem>", GTK_STOCK_PREFERENCES },
{ N_("/Tools/Pr_ivacy"), NULL, gaim_gtk_privacy_dialog_show, 0, "<StockItem>", GTK_STOCK_DIALOG_ERROR },
{ "/Tools/sep2", NULL, NULL, 0, "<Separator>", NULL },
{ N_("/Tools/_File Transfers"), "<CTL>T", gaim_gtkxfer_dialog_show, 0, "<StockItem>", GAIM_STOCK_FILE_TRANSFER },
{ N_("/Tools/R_oom List"), NULL, gaim_gtk_roomlist_dialog_show, 0, "<StockItem>", GTK_STOCK_INDEX },
{ N_("/Tools/System _Log"), NULL, gtk_blist_show_systemlog_cb, 0, "<StockItem>", GAIM_STOCK_LOG },
{ "/Tools/sep3", NULL, NULL, 0, "<Separator>", NULL },
{ N_("/Tools/Mute _Sounds"), "<CTL>S", gaim_gtk_blist_mute_sounds_cb, 0, "<CheckItem>", NULL },
/* Help */
{ N_("/_Help"), NULL, NULL, 0, "<Branch>", NULL },
{ N_("/Help/Online _Help"), "F1", gtk_blist_show_onlinehelp_cb, 0, "<StockItem>", GTK_STOCK_HELP },
{ N_("/Help/_Debug Window"), NULL, toggle_debug, 0, "<StockItem>", GAIM_STOCK_DEBUG },
{ N_("/Help/_About"), NULL, gaim_gtkdialogs_about, 0, "<StockItem>", GAIM_STOCK_ABOUT },
};
/*********************************************************
* Private Utility functions *
*********************************************************/
static char *gaim_get_tooltip_text(GaimBlistNode *node, gboolean full)
{
GString *str = g_string_new("");
GaimPlugin *prpl;
GaimPluginProtocolInfo *prpl_info = NULL;
char *tmp;
if (GAIM_BLIST_NODE_IS_CHAT(node))
{
GaimChat *chat;
GList *cur;
struct proto_chat_entry *pce;
char *name, *value;
chat = (GaimChat *)node;
prpl = gaim_find_prpl(gaim_account_get_protocol_id(chat->account));
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
tmp = g_markup_escape_text(gaim_chat_get_name(chat), -1);
g_string_append_printf(str, "<span size='larger' weight='bold'>%s</span>", tmp);
g_free(tmp);
if (g_list_length(gaim_connections_get_all()) > 1)
{
tmp = g_markup_escape_text(chat->account->username, -1);
g_string_append_printf(str, _("\n<b>Account:</b> %s"), tmp);
g_free(tmp);
}
if (prpl_info->chat_info != NULL)
cur = prpl_info->chat_info(chat->account->gc);
else
cur = NULL;
while (cur != NULL)
{
pce = cur->data;
if (!pce->secret && (!pce->required &&
g_hash_table_lookup(chat->components, pce->identifier) == NULL))
{
tmp = gaim_text_strip_mnemonic(pce->label);
name = g_markup_escape_text(tmp, -1);
g_free(tmp);
value = g_markup_escape_text(g_hash_table_lookup(
chat->components, pce->identifier), -1);
g_string_append_printf(str, "\n<b>%s</b> %s",
name ? name : "",
value ? value : "");
g_free(name);
g_free(value);
}
g_free(pce);
cur = g_list_remove(cur, pce);
}
}
else if (GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_BUDDY(node))
{
/* NOTE: THIS FUNCTION IS NO LONGER CALLED FOR CONTACTS.
* It is only called by create_tip_for_node(), and create_tip_for_node() is never called for a contact.
*/
GaimContact *c;
GaimBuddy *b;
GaimPresence *presence;
GaimNotifyUserInfo *user_info;
char *tmp;
time_t idle_secs, signon;
if (GAIM_BLIST_NODE_IS_CONTACT(node))
{
c = (GaimContact *)node;
b = gaim_contact_get_priority_buddy(c);
}
else
{
b = (GaimBuddy *)node;
c = gaim_buddy_get_contact(b);
}
prpl = gaim_find_prpl(gaim_account_get_protocol_id(b->account));
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
presence = gaim_buddy_get_presence(b);
/* Buddy Name */
tmp = g_markup_escape_text(gaim_buddy_get_name(b), -1);
g_string_append_printf(str, "<span size='larger' weight='bold'>%s</span>\n", tmp);
g_free(tmp);
user_info = gaim_notify_user_info_new();
/* Account */
if (full && g_list_length(gaim_connections_get_all()) > 1)
{
tmp = g_markup_escape_text(gaim_account_get_username(
gaim_buddy_get_account(b)), -1);
gaim_notify_user_info_add_pair(user_info, _("Account"), tmp);
g_free(tmp);
}
/* Alias */
/* If there's not a contact alias, the node is being displayed with
* this alias, so there's no point in showing it in the tooltip. */
if (full && b->alias != NULL && b->alias[0] != '\0' &&
(c->alias != NULL && c->alias[0] != '\0') &&
strcmp(c->alias, b->alias) != 0)
{
tmp = g_markup_escape_text(b->alias, -1);
gaim_notify_user_info_add_pair(user_info, _("Buddy Alias"), tmp);
g_free(tmp);
}
/* Nickname/Server Alias */
/* I'd like to only show this if there's a contact or buddy
* alias, but many people on MSN set long nicknames, which
* get ellipsized, so the only way to see the whole thing is
* to look at the tooltip. */
if (full && b->server_alias != NULL && b->server_alias[0] != '\0')
{
tmp = g_markup_escape_text(b->server_alias, -1);
gaim_notify_user_info_add_pair(user_info, _("Nickname"), tmp);
g_free(tmp);
}
/* Logged In */
signon = gaim_presence_get_login_time(presence);
if (full && GAIM_BUDDY_IS_ONLINE(b) && signon > 0)
{
tmp = gaim_str_seconds_to_string(time(NULL) - signon);
gaim_notify_user_info_add_pair(user_info, _("Logged In"), tmp);
g_free(tmp);
}
/* Idle */
if (gaim_presence_is_idle(presence))
{
idle_secs = gaim_presence_get_idle_time(presence);
if (idle_secs > 0)
{
tmp = gaim_str_seconds_to_string(time(NULL) - idle_secs);
gaim_notify_user_info_add_pair(user_info, _("Idle"), tmp);
g_free(tmp);
}
}
/* Last Seen */
if (full && !GAIM_BUDDY_IS_ONLINE(b))
{
struct _gaim_gtk_blist_node *gtknode = ((GaimBlistNode *)c)->ui_data;
GaimBlistNode *bnode;
int lastseen = 0;
if (!gtknode->contact_expanded || GAIM_BLIST_NODE_IS_CONTACT(node))
{
/* We're either looking at a buddy for a collapsed contact or
* an expanded contact itself so we show the most recent
* (largest) last_seen time for any of the buddies under
* the contact. */
for (bnode = ((GaimBlistNode *)c)->child ; bnode != NULL ; bnode = bnode->next)
{
int value = gaim_blist_node_get_int(bnode, "last_seen");
if (value > lastseen)
lastseen = value;
}
}
else
{
/* We're dealing with a buddy under an expanded contact,
* so we show the last_seen time for the buddy. */
lastseen = gaim_blist_node_get_int(&b->node, "last_seen");
}
if (lastseen > 0)
{
tmp = gaim_str_seconds_to_string(time(NULL) - lastseen);
gaim_notify_user_info_add_pair(user_info, _("Last Seen"), tmp);
g_free(tmp);
}
}
/* Offline? */
/* FIXME: Why is this status special-cased by the core? -- rlaager */
if (!GAIM_BUDDY_IS_ONLINE(b)) {
gaim_notify_user_info_add_pair(user_info, _("Status"), _("Offline"));
}
if (prpl_info && prpl_info->tooltip_text)
{
/* Additional text from the PRPL */
prpl_info->tooltip_text(b, user_info, full);
}
/* These are Easter Eggs. Patches to remove them will be rejected. */
if (!g_ascii_strcasecmp(b->name, "robflynn"))
gaim_notify_user_info_add_pair(user_info, _("Description"), _("Spooky"));
if (!g_ascii_strcasecmp(b->name, "seanegn"))
gaim_notify_user_info_add_pair(user_info, _("Status"), _("Awesome"));
if (!g_ascii_strcasecmp(b->name, "chipx86"))
gaim_notify_user_info_add_pair(user_info, _("Status"), _("Rockin'"));
tmp = gaim_notify_user_info_get_text_with_newline(user_info, "\n");
g_string_append(str, tmp);
g_free(tmp);
gaim_notify_user_info_destroy(user_info);
}
gaim_signal_emit(gaim_gtk_blist_get_handle(),
"drawing-tooltip", node, str, full);
return g_string_free(str, FALSE);
}
struct _emblem_data {
const char *filename;
int x;
int y;
};
static void g_string_destroy(GString *destroyable)
{
g_string_free(destroyable, TRUE);
}
static void
gaim_gtk_blist_update_buddy_status_icon_key(struct _gaim_gtk_blist_node *gtkbuddynode, GaimBuddy *buddy, GaimStatusIconSize size)
{
GString *key = g_string_sized_new(16);
if(gtkbuddynode && gtkbuddynode->recent_signonoff) {
if(GAIM_BUDDY_IS_ONLINE(buddy))
g_string_printf(key, "login");
else
g_string_printf(key, "logout");
} else {
int i;
const char *protoname = NULL;
GaimAccount *account = buddy->account;
GaimPlugin *prpl = gaim_find_prpl(gaim_account_get_protocol_id(account));
GaimPluginProtocolInfo *prpl_info;
GaimConversation *conv;
struct _emblem_data emblems[4] = {{NULL, 15, 15}, {NULL, 0, 15},
{NULL, 0, 0}, {NULL, 15, 0}};
if(!prpl)
return;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if(prpl_info && prpl_info->list_icon) {
protoname = prpl_info->list_icon(account, buddy);
}
if(prpl_info && prpl_info->list_emblems) {
prpl_info->list_emblems(buddy, &emblems[0].filename,
&emblems[1].filename, &emblems[2].filename,
&emblems[3].filename);
}
g_string_assign(key, protoname);
conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM,
gaim_buddy_get_name(buddy), account);
if(conv != NULL) {
GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
if(gaim_gtkconv_is_hidden(gtkconv)) {
/* add pending emblem */
if(size == GAIM_STATUS_ICON_SMALL) {
emblems[0].filename = "pending";
}
else {
emblems[3].filename = emblems[2].filename;
emblems[2].filename = "pending";
}
}
}
if(size == GAIM_STATUS_ICON_SMALL) {
/* So that only the se icon will composite */
emblems[1].filename = emblems[2].filename = emblems[3].filename = NULL;
}
for(i = 0; i < 4; i++) {
if(emblems[i].filename)
g_string_append_printf(key, "/%s", emblems[i].filename);
}
}
if (!GAIM_BUDDY_IS_ONLINE(buddy))
g_string_append(key, "/off");
else if (gaim_presence_is_idle(gaim_buddy_get_presence(buddy)))
g_string_append(key, "/idle");
if (!gaim_privacy_check(buddy->account, gaim_buddy_get_name(buddy)))
g_string_append(key, "/priv");
if (gtkbuddynode->status_icon_key)
g_string_free(gtkbuddynode->status_icon_key, TRUE);
gtkbuddynode->status_icon_key = key;
}
GdkPixbuf *
gaim_gtk_blist_get_status_icon(GaimBlistNode *node, GaimStatusIconSize size)
{
GdkPixbuf *scale, *status = NULL;
int i, scalesize = 30;
char *filename;
GString *key = g_string_sized_new(16);
const char *protoname = NULL;
struct _gaim_gtk_blist_node *gtknode = node->ui_data;
struct _gaim_gtk_blist_node *gtkbuddynode = NULL;
struct _emblem_data emblems[4] = {{NULL, 15, 15}, {NULL, 0, 15},
{NULL, 0, 0}, {NULL, 15, 0}};
GaimPresence *presence = NULL;
GaimBuddy *buddy = NULL;
GaimChat *chat = NULL;
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
if(!gtknode->contact_expanded) {
buddy = gaim_contact_get_priority_buddy((GaimContact*)node);
gtkbuddynode = ((GaimBlistNode*)buddy)->ui_data;
}
} else if(GAIM_BLIST_NODE_IS_BUDDY(node)) {
buddy = (GaimBuddy*)node;
gtkbuddynode = node->ui_data;
} else if(GAIM_BLIST_NODE_IS_CHAT(node)) {
chat = (GaimChat*)node;
} else {
return NULL;
}
if (!status_icon_hash_table) {
status_icon_hash_table = g_hash_table_new_full((GHashFunc)g_string_hash,
(GEqualFunc)g_string_equal,
(GDestroyNotify)g_string_destroy,
(GDestroyNotify)gdk_pixbuf_unref);
} else if (buddy && gtkbuddynode->status_icon_key && gtkbuddynode->status_icon_key->str) {
key = g_string_new(gtkbuddynode->status_icon_key->str);
/* Respect the size request given */
if (size == GAIM_STATUS_ICON_SMALL) {
g_string_append(key, "/tiny");
}
scale = g_hash_table_lookup(status_icon_hash_table, key);
if (scale) {
gdk_pixbuf_ref(scale);
g_string_free(key, TRUE);
return scale;
}
}
if(buddy || chat) {
GaimAccount *account;
GaimPlugin *prpl;
GaimPluginProtocolInfo *prpl_info;
if(buddy)
account = buddy->account;
else
account = chat->account;
prpl = gaim_find_prpl(gaim_account_get_protocol_id(account));
if(!prpl)
return NULL;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if(prpl_info && prpl_info->list_icon) {
protoname = prpl_info->list_icon(account, buddy);
}
if(prpl_info && prpl_info->list_emblems && buddy) {
if(gtknode && !gtknode->recent_signonoff)
prpl_info->list_emblems(buddy, &emblems[0].filename,
&emblems[1].filename, &emblems[2].filename,
&emblems[3].filename);
}
}
/* Begin Generating Lookup Key */
if (buddy) {
gaim_gtk_blist_update_buddy_status_icon_key(gtkbuddynode, buddy, size);
g_string_assign(key, gtkbuddynode->status_icon_key->str);
}
/* There are only two options for chat or gaimdude - big or small */
else if (chat) {
GaimAccount *account;
GaimPlugin *prpl;
GaimPluginProtocolInfo *prpl_info;
account = chat->account;
prpl = gaim_find_prpl(gaim_account_get_protocol_id(account));
if(!prpl)
return NULL;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if(prpl_info && prpl_info->list_icon) {
protoname = prpl_info->list_icon(account, NULL);
}
g_string_append_printf(key, "%s-chat", protoname);
}
else
g_string_append(key, "gaimdude");
/* If the icon is small, we do not store this into the status_icon_key
* in the gtkbuddynode. This way we can respect the size value on cache
* lookup. Otherwise, different sized icons could not be stored easily.
*/
if (size == GAIM_STATUS_ICON_SMALL) {
g_string_append(key, "/tiny");
}
/* End Generating Lookup Key */
/* If we already know this icon, just return it */
scale = g_hash_table_lookup(status_icon_hash_table, key);
if (scale) {
gdk_pixbuf_ref(scale);
g_string_free(key, TRUE);
return scale;
}
/* Create a new composite icon */
if(buddy) {
GaimAccount *account;
GaimPlugin *prpl;
GaimPluginProtocolInfo *prpl_info;
GaimConversation *conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM,
gaim_buddy_get_name(buddy),
gaim_buddy_get_account(buddy));
account = buddy->account;
prpl = gaim_find_prpl(gaim_account_get_protocol_id(account));
if(!prpl)
return NULL;
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if(prpl_info && prpl_info->list_icon) {
protoname = prpl_info->list_icon(account, buddy);
}
if(prpl_info && prpl_info->list_emblems) {
if(gtknode && !gtknode->recent_signonoff)
prpl_info->list_emblems(buddy, &emblems[0].filename,
&emblems[1].filename, &emblems[2].filename,
&emblems[3].filename);
}
if(conv != NULL) {
GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
if(gtkconv != NULL && gaim_gtkconv_is_hidden(gtkconv)) {
/* add pending emblem */
if(size == GAIM_STATUS_ICON_SMALL) {
emblems[0].filename="pending";
}
else {
emblems[3].filename=emblems[2].filename;
emblems[2].filename="pending";
}
}
}
}
if(size == GAIM_STATUS_ICON_SMALL) {
scalesize = 15;
/* So that only the se icon will composite */
emblems[1].filename = emblems[2].filename = emblems[3].filename = NULL;
}
if(buddy && GAIM_BUDDY_IS_ONLINE(buddy) && gtkbuddynode && gtkbuddynode->recent_signonoff) {
filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", "login.png", NULL);
} else if(buddy && !GAIM_BUDDY_IS_ONLINE(buddy) && gtkbuddynode && gtkbuddynode->recent_signonoff) {
filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", "logout.png", NULL);
} else if(buddy || chat) {
char *image = g_strdup_printf("%s.png", protoname);
filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", image, NULL);
g_free(image);
} else {
/* gaim dude */
filename = g_build_filename(DATADIR, "pixmaps", "gaim.png", NULL);
}
status = gdk_pixbuf_new_from_file(filename, NULL);
g_free(filename);
if(!status) {
g_string_free(key, TRUE);
return NULL;
}
scale = gdk_pixbuf_scale_simple(status, scalesize, scalesize,
GDK_INTERP_BILINEAR);
g_object_unref(status);
for(i=0; i<4; i++) {
if(emblems[i].filename) {
GdkPixbuf *emblem;
char *image = g_strdup_printf("%s.png", emblems[i].filename);
filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", image, NULL);
g_free(image);
emblem = gdk_pixbuf_new_from_file(filename, NULL);
g_free(filename);
if(emblem) {
if(i == 0 && size == GAIM_STATUS_ICON_SMALL) {
double scale_factor = 0.6;
if(gdk_pixbuf_get_width(emblem) > 20)
scale_factor = 9.0 / gdk_pixbuf_get_width(emblem);
gdk_pixbuf_composite(emblem,
scale, 5, 5,
10, 10,
5, 5,
scale_factor, scale_factor,
GDK_INTERP_BILINEAR,
255);
} else {
double scale_factor = 1.0;
if(gdk_pixbuf_get_width(emblem) > 20)
scale_factor = 15.0 / gdk_pixbuf_get_width(emblem);
gdk_pixbuf_composite(emblem,
scale, emblems[i].x, emblems[i].y,
15, 15,
emblems[i].x, emblems[i].y,
scale_factor, scale_factor,
GDK_INTERP_BILINEAR,
255);
}
g_object_unref(emblem);
}
}
}
if(buddy) {
presence = gaim_buddy_get_presence(buddy);
if (!GAIM_BUDDY_IS_ONLINE(buddy))
gdk_pixbuf_saturate_and_pixelate(scale, scale, 0.0, FALSE);
else if (gaim_presence_is_idle(presence))
{
gdk_pixbuf_saturate_and_pixelate(scale, scale, 0.25, FALSE);
}
if (!gaim_privacy_check(buddy->account, gaim_buddy_get_name(buddy)))
{
GdkPixbuf *emblem;
char *filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", "blocked.png", NULL);
emblem = gdk_pixbuf_new_from_file(filename, NULL);
g_free(filename);
if (emblem)
{
gdk_pixbuf_composite(emblem, scale,
0, 0, scalesize, scalesize,
0, 0,
(double)scalesize / gdk_pixbuf_get_width(emblem),
(double)scalesize / gdk_pixbuf_get_height(emblem),
GDK_INTERP_BILINEAR,
224);
g_object_unref(emblem);
}
}
}
/* Insert the new icon into the status icon hash table */
g_hash_table_insert (status_icon_hash_table, key, scale);
gdk_pixbuf_ref(scale);
return scale;
}
static gchar *gaim_gtk_blist_get_name_markup(GaimBuddy *b, gboolean selected)
{
const char *name;
char *esc, *text = NULL;
GaimPlugin *prpl;
GaimPluginProtocolInfo *prpl_info = NULL;
GaimContact *contact;
GaimPresence *presence;
struct _gaim_gtk_blist_node *gtkcontactnode = NULL;
char *idletime = NULL, *statustext = NULL;
time_t t;
/* XXX Good luck cleaning up this crap */
contact = (GaimContact*)((GaimBlistNode*)b)->parent;
if(contact)
gtkcontactnode = ((GaimBlistNode*)contact)->ui_data;
if(gtkcontactnode && !gtkcontactnode->contact_expanded && contact->alias)
name = contact->alias;
else
name = gaim_buddy_get_alias(b);
esc = g_markup_escape_text(name, strlen(name));
presence = gaim_buddy_get_presence(b);
if (!gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"))
{
if (!selected && gaim_presence_is_idle(presence))
{
text = g_strdup_printf("<span color='%s'>%s</span>",
dim_grey(), esc);
g_free(esc);
return text;
}
else
return esc;
}
prpl = gaim_find_prpl(gaim_account_get_protocol_id(b->account));
if (prpl != NULL)
prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(prpl);
if (prpl_info && prpl_info->status_text && b->account->gc) {
char *tmp = prpl_info->status_text(b);
const char *end;
if(tmp && !g_utf8_validate(tmp, -1, &end)) {
char *new = g_strndup(tmp,
g_utf8_pointer_to_offset(tmp, end));
g_free(tmp);
tmp = new;
}
#if !GTK_CHECK_VERSION(2,6,0)
if(tmp) {
char buf[32];
char *c = tmp;
int length = 0, vis=0;
gboolean inside = FALSE;
g_strdelimit(tmp, "\n", ' ');
gaim_str_strip_char(tmp, '\r');
while(*c && vis < 20) {
if(*c == '&')
inside = TRUE;
else if(*c == ';')
inside = FALSE;
if(!inside)
vis++;
c = g_utf8_next_char(c); /* this is fun */
}
length = c - tmp;
if(vis == 20)
g_snprintf(buf, sizeof(buf), "%%.%ds...", length);
else
g_snprintf(buf, sizeof(buf), "%%s ");
statustext = g_strdup_printf(buf, tmp);
g_free(tmp);
}
#else
if(tmp) {
g_strdelimit(tmp, "\n", ' ');
gaim_str_strip_char(tmp, '\r');
}
statustext = tmp;
#endif
}
if(!gaim_presence_is_online(presence) && !statustext)
statustext = g_strdup(_("Offline"));
else if (!statustext)
text = g_strdup(esc);
if (gaim_presence_is_idle(presence)) {
if (gaim_prefs_get_bool("/gaim/gtk/blist/show_idle_time")) {
time_t idle_secs = gaim_presence_get_idle_time(presence);
if (idle_secs > 0) {
int ihrs, imin;
time(&t);
ihrs = (t - idle_secs) / 3600;
imin = ((t - idle_secs) / 60) % 60;
if (ihrs)
idletime = g_strdup_printf(_("Idle %dh %02dm"), ihrs, imin);
else
idletime = g_strdup_printf(_("Idle %dm"), imin);
}
else
idletime = g_strdup(_("Idle"));
if (!selected)
text = g_strdup_printf("<span color='%s'>%s</span>\n"
"<span color='%s' size='smaller'>%s%s%s</span>",
dim_grey(), esc, dim_grey(),
idletime != NULL ? idletime : "",
(idletime != NULL && statustext != NULL) ? " - " : "",
statustext != NULL ? statustext : "");
}
else if (!selected && !statustext) /* We handle selected text later */
text = g_strdup_printf("<span color='%s'>%s</span>", dim_grey(), esc);
else if (!selected && !text)
text = g_strdup_printf("<span color='%s'>%s</span>\n"
"<span color='%s' size='smaller'>%s</span>",
dim_grey(), esc, dim_grey(),
statustext != NULL ? statustext : "");
}
/* Not idle and not selected */
else if (!selected && !text)
{
text = g_strdup_printf("%s\n"
"<span color='%s' size='smaller'>%s</span>",
esc, dim_grey(),
statustext != NULL ? statustext : "");
}
/* It is selected. */
if ((selected && !text) || (selected && idletime))
text = g_strdup_printf("%s\n"
"<span size='smaller'>%s%s%s</span>",
esc,
idletime != NULL ? idletime : "",
(idletime != NULL && statustext != NULL) ? " - " : "",
statustext != NULL ? statustext : "");
g_free(idletime);
g_free(statustext);
g_free(esc);
return text;
}
static void gaim_gtk_blist_restore_position()
{
int blist_x, blist_y, blist_width, blist_height;
blist_width = gaim_prefs_get_int("/gaim/gtk/blist/width");
/* if the window exists, is hidden, we're saving positions, and the
* position is sane... */
if (gtkblist && gtkblist->window &&
!GTK_WIDGET_VISIBLE(gtkblist->window) && blist_width != 0) {
blist_x = gaim_prefs_get_int("/gaim/gtk/blist/x");
blist_y = gaim_prefs_get_int("/gaim/gtk/blist/y");
blist_height = gaim_prefs_get_int("/gaim/gtk/blist/height");
/* ...check position is on screen... */
if (blist_x >= gdk_screen_width())
blist_x = gdk_screen_width() - 100;
else if (blist_x + blist_width < 0)
blist_x = 100;
if (blist_y >= gdk_screen_height())
blist_y = gdk_screen_height() - 100;
else if (blist_y + blist_height < 0)
blist_y = 100;
/* ...and move it back. */
gtk_window_move(GTK_WINDOW(gtkblist->window), blist_x, blist_y);
gtk_window_resize(GTK_WINDOW(gtkblist->window), blist_width, blist_height);
if (gaim_prefs_get_bool("/gaim/gtk/blist/list_maximized"))
gtk_window_maximize(GTK_WINDOW(gtkblist->window));
}
}
static gboolean gaim_gtk_blist_refresh_timer(GaimBuddyList *list)
{
GaimBlistNode *gnode, *cnode;
if (gtk_blist_obscured || !GTK_WIDGET_VISIBLE(gtkblist->window))
return TRUE;
for(gnode = list->root; gnode; gnode = gnode->next) {
if(!GAIM_BLIST_NODE_IS_GROUP(gnode))
continue;
for(cnode = gnode->child; cnode; cnode = cnode->next) {
if(GAIM_BLIST_NODE_IS_CONTACT(cnode)) {
GaimBuddy *buddy;
buddy = gaim_contact_get_priority_buddy((GaimContact*)cnode);
if (buddy &&
gaim_presence_is_idle(gaim_buddy_get_presence(buddy)))
gaim_gtk_blist_update_contact(list, (GaimBlistNode*)buddy);
}
}
}
/* keep on going */
return TRUE;
}
static void gaim_gtk_blist_hide_node(GaimBuddyList *list, GaimBlistNode *node, gboolean update)
{
struct _gaim_gtk_blist_node *gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
GtkTreeIter iter;
if (!gtknode || !gtknode->row || !gtkblist)
return;
if(gtkblist->selected_node == node)
gtkblist->selected_node = NULL;
if (get_iter_from_node(node, &iter)) {
gtk_tree_store_remove(gtkblist->treemodel, &iter);
if(update && (GAIM_BLIST_NODE_IS_CONTACT(node) ||
GAIM_BLIST_NODE_IS_BUDDY(node) || GAIM_BLIST_NODE_IS_CHAT(node))) {
gaim_gtk_blist_update(list, node->parent);
}
}
gtk_tree_row_reference_free(gtknode->row);
gtknode->row = NULL;
}
static const char *require_connection[] =
{
N_("/Buddies/New Instant Message..."),
N_("/Buddies/Join a Chat..."),
N_("/Buddies/Get User Info..."),
N_("/Buddies/Add Buddy..."),
N_("/Buddies/Add Chat..."),
N_("/Buddies/Add Group..."),
};
static const int require_connection_size = sizeof(require_connection)
/ sizeof(*require_connection);
/**
* Rebuild dynamic menus and make menu items sensitive/insensitive
* where appropriate.
*/
static void
update_menu_bar(GaimGtkBuddyList *gtkblist)
{
GtkWidget *widget;
gboolean sensitive;
int i;
g_return_if_fail(gtkblist != NULL);
gaim_gtk_blist_update_accounts_menu();
sensitive = (gaim_connections_get_all() != NULL);
for (i = 0; i < require_connection_size; i++)
{
widget = gtk_item_factory_get_widget(gtkblist->ift, require_connection[i]);
gtk_widget_set_sensitive(widget, sensitive);
}
widget = gtk_item_factory_get_widget(gtkblist->ift, N_("/Buddies/Join a Chat..."));
gtk_widget_set_sensitive(widget, gaim_gtk_blist_joinchat_is_showable());
widget = gtk_item_factory_get_widget(gtkblist->ift, N_("/Buddies/Add Chat..."));
gtk_widget_set_sensitive(widget, gaim_gtk_blist_joinchat_is_showable());
widget = gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools/Buddy Pounces"));
gtk_widget_set_sensitive(widget, (gaim_accounts_get_all() != NULL));
widget = gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools/Privacy"));
gtk_widget_set_sensitive(widget, (gaim_connections_get_all() != NULL));
widget = gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools/Room List"));
gtk_widget_set_sensitive(widget, gaim_gtk_roomlist_is_showable());
}
static void
sign_on_off_cb(GaimConnection *gc, GaimBuddyList *blist)
{
GaimGtkBuddyList *gtkblist = GAIM_GTK_BLIST(blist);
update_menu_bar(gtkblist);
}
static void
plugin_changed_cb(GaimPlugin *p, gpointer *data)
{
gaim_gtk_blist_update_plugin_actions();
}
static void
unseen_conv_menu()
{
static GtkWidget *menu = NULL;
GList *convs = NULL;
if (menu) {
gtk_widget_destroy(menu);
menu = NULL;
}
convs = gaim_gtk_conversations_find_unseen_list(GAIM_CONV_TYPE_IM, GAIM_UNSEEN_TEXT, TRUE, 0);
if (!convs)
/* no conversations added, don't show the menu */
return;
menu = gtk_menu_new();
gaim_gtk_conversations_fill_menu(menu, convs);
g_list_free(convs);
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 3,
gtk_get_current_event_time());
}
static gboolean
menutray_press_cb(GtkWidget *widget, GdkEventButton *event)
{
GList *convs;
switch (event->button) {
case 1:
convs = gaim_gtk_conversations_find_unseen_list(GAIM_CONV_TYPE_IM,
GAIM_UNSEEN_TEXT, TRUE, 1);
if (convs) {
gaim_gtkconv_present_conversation((GaimConversation*)convs->data);
g_list_free(convs);
}
break;
case 3:
unseen_conv_menu();
break;
}
return TRUE;
}
static void
conversation_updated_cb(GaimConversation *conv, GaimConvUpdateType type,
GaimGtkBuddyList *gtkblist)
{
GList *convs = NULL;
GList *l = NULL;
if (type != GAIM_CONV_UPDATE_UNSEEN)
return;
if(conv->account != NULL && conv->name != NULL) {
GaimBuddy *buddy = gaim_find_buddy(conv->account, conv->name);
if(buddy != NULL)
gaim_gtk_blist_update_buddy(NULL, (GaimBlistNode *)buddy, TRUE);
}
if (gtkblist->menutrayicon) {
gtk_widget_destroy(gtkblist->menutrayicon);
gtkblist->menutrayicon = NULL;
}
convs = gaim_gtk_conversations_find_unseen_list(GAIM_CONV_TYPE_IM, GAIM_UNSEEN_TEXT, TRUE, 0);
if (convs) {
GtkWidget *img = NULL;
GString *tooltip_text = NULL;
tooltip_text = g_string_new("");
l = convs;
while (l != NULL) {
if (GAIM_IS_GTK_CONVERSATION(l->data)) {
GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION((GaimConversation *)l->data);
g_string_append_printf(tooltip_text,
ngettext("%d unread message from %s\n", "%d unread messages from %s\n", gtkconv->unseen_count),
gtkconv->unseen_count,
gtk_label_get_text(GTK_LABEL(gtkconv->tab_label)));
}
l = l->next;
}
if(tooltip_text->len > 0) {
/* get rid of the last newline */
g_string_truncate(tooltip_text, tooltip_text->len -1);
img = gtk_image_new_from_stock(GAIM_STOCK_PENDING, GTK_ICON_SIZE_MENU);
gtkblist->menutrayicon = gtk_event_box_new();
gtk_container_add(GTK_CONTAINER(gtkblist->menutrayicon), img);
gtk_widget_show(img);
gtk_widget_show(gtkblist->menutrayicon);
g_signal_connect(G_OBJECT(gtkblist->menutrayicon), "button-press-event", G_CALLBACK(menutray_press_cb), NULL);
gaim_gtk_menu_tray_append(GAIM_GTK_MENU_TRAY(gtkblist->menutray), gtkblist->menutrayicon, tooltip_text->str);
}
g_string_free(tooltip_text, TRUE);
g_list_free(convs);
}
}
static void
conversation_deleting_cb(GaimConversation *conv, GaimGtkBuddyList *gtkblist)
{
conversation_updated_cb(conv, GAIM_CONV_UPDATE_UNSEEN, gtkblist);
}
/**********************************************************************************
* Public API Functions *
**********************************************************************************/
static void gaim_gtk_blist_new_list(GaimBuddyList *blist)
{
GaimGtkBuddyList *gtkblist;
gtkblist = g_new0(GaimGtkBuddyList, 1);
gtkblist->connection_errors = g_hash_table_new_full(g_direct_hash,
g_direct_equal, NULL, g_free);
blist->ui_data = gtkblist;
}
static void gaim_gtk_blist_new_node(GaimBlistNode *node)
{
node->ui_data = g_new0(struct _gaim_gtk_blist_node, 1);
}
gboolean gaim_gtk_blist_node_is_contact_expanded(GaimBlistNode *node)
{
if GAIM_BLIST_NODE_IS_BUDDY(node)
node = node->parent;
g_return_val_if_fail(GAIM_BLIST_NODE_IS_CONTACT(node), FALSE);
return ((struct _gaim_gtk_blist_node *)node->ui_data)->contact_expanded;
}
enum {
DRAG_BUDDY,
DRAG_ROW,
DRAG_VCARD,
DRAG_TEXT,
DRAG_URI,
NUM_TARGETS
};
static const char *
item_factory_translate_func (const char *path, gpointer func_data)
{
return _((char *)path);
}
void gaim_gtk_blist_setup_sort_methods()
{
gaim_gtk_blist_sort_method_reg("none", _("Manually"), sort_method_none);
#if GTK_CHECK_VERSION(2,2,1)
gaim_gtk_blist_sort_method_reg("alphabetical", _("Alphabetically"), sort_method_alphabetical);
gaim_gtk_blist_sort_method_reg("status", _("By status"), sort_method_status);
gaim_gtk_blist_sort_method_reg("log_size", _("By log size"), sort_method_log);
#endif
gaim_gtk_blist_sort_method_set(gaim_prefs_get_string("/gaim/gtk/blist/sort_type"));
}
static void _prefs_change_redo_list()
{
GtkTreeSelection *sel;
GtkTreeIter iter;
GaimBlistNode *node = NULL;
sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkblist->treeview));
if (gtk_tree_selection_get_selected(sel, NULL, &iter))
{
gtk_tree_model_get(GTK_TREE_MODEL(gtkblist->treemodel), &iter, NODE_COLUMN, &node, -1);
}
redo_buddy_list(gaim_get_blist(), FALSE, FALSE);
#if GTK_CHECK_VERSION(2,6,0)
gtk_tree_view_columns_autosize(GTK_TREE_VIEW(gtkblist->treeview));
#endif
if (node)
{
struct _gaim_gtk_blist_node *gtknode;
GtkTreePath *path;
gtknode = node->ui_data;
if (gtknode && gtknode->row)
{
path = gtk_tree_row_reference_get_path(gtknode->row);
gtk_tree_selection_select_path(sel, path);
gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(gtkblist->treeview), path, NULL, FALSE, 0, 0);
gtk_tree_path_free(path);
}
}
}
static void _prefs_change_sort_method(const char *pref_name, GaimPrefType type,
gconstpointer val, gpointer data)
{
if(!strcmp(pref_name, "/gaim/gtk/blist/sort_type"))
gaim_gtk_blist_sort_method_set(val);
}
static void account_modified(GaimAccount *account, GaimGtkBuddyList *gtkblist)
{
GList *list;
if (!gtkblist)
return;
if ((list = gaim_accounts_get_all_active()) != NULL) {
gtk_notebook_set_current_page(GTK_NOTEBOOK(gtkblist->notebook), 1);
g_list_free(list);
} else
gtk_notebook_set_current_page(GTK_NOTEBOOK(gtkblist->notebook), 0);
update_menu_bar(gtkblist);
}
static void
account_status_changed(GaimAccount *account, GaimStatus *old,
GaimStatus *new, GaimGtkBuddyList *gtkblist)
{
if (!gtkblist)
return;
update_menu_bar(gtkblist);
}
static gboolean
gtk_blist_window_key_press_cb(GtkWidget *w, GdkEventKey *event, GaimGtkBuddyList *gtkblist)
{
GtkWidget *imhtml;
if (!gtkblist)
return FALSE;
imhtml = gtk_window_get_focus(GTK_WINDOW(gtkblist->window));
if (GTK_IS_IMHTML(imhtml) && gtk_bindings_activate(GTK_OBJECT(imhtml), event->keyval, event->state))
return TRUE;
return FALSE;
}
static gboolean
headline_hover_close(int x, int y)
{
GtkWidget *w = gtkblist->headline_hbox;
if (x <= w->allocation.width && x >= w->allocation.width - HEADLINE_CLOSE_SIZE &&
y >= 0 && y <= HEADLINE_CLOSE_SIZE)
return TRUE;
return FALSE;
}
static gboolean
headline_box_enter_cb(GtkWidget *widget, GdkEventCrossing *event, GaimGtkBuddyList *gtkblist)
{
gdk_window_set_cursor(widget->window, gtkblist->hand_cursor);
if (gtkblist->headline_close) {
#if GTK_CHECK_VERSION(2,2,0)
gdk_draw_pixbuf(widget->window, NULL, gtkblist->headline_close,
#else
gdk_pixbuf_render_to_drawable(gtkblist->headline_close,
GDK_DRAWABLE(widget->window), NULL,
#endif
0, 0,
widget->allocation.width - 2 - HEADLINE_CLOSE_SIZE, 2,
HEADLINE_CLOSE_SIZE, HEADLINE_CLOSE_SIZE,
GDK_RGB_DITHER_NONE, 0, 0);
gtk_paint_focus(widget->style, widget->window, GTK_STATE_PRELIGHT,
NULL, widget, NULL,
widget->allocation.width - HEADLINE_CLOSE_SIZE - 3, 1,
HEADLINE_CLOSE_SIZE + 2, HEADLINE_CLOSE_SIZE + 2);
}
return FALSE;
}
#if 0
static gboolean
headline_box_motion_cb(GtkWidget *widget, GdkEventMotion *event, GaimGtkBuddyList *gtkblist)
{
gaim_debug_fatal("motion", "%d %d\n", (int)event->x, (int)event->y);
if (headline_hover_close((int)event->x, (int)event->y))
gtk_paint_focus(widget->style, widget->window, GTK_STATE_PRELIGHT,
NULL, widget, NULL,
widget->allocation.width - HEADLINE_CLOSE_SIZE - 3, 1,
HEADLINE_CLOSE_SIZE + 2, HEADLINE_CLOSE_SIZE + 2);
return FALSE;
}
#endif
static gboolean
headline_box_leave_cb(GtkWidget *widget, GdkEventCrossing *event, GaimGtkBuddyList *gtkblist)
{
gdk_window_set_cursor(widget->window, gtkblist->arrow_cursor);
if (gtkblist->headline_close) {
GdkRectangle rect = {widget->allocation.width - 3 - HEADLINE_CLOSE_SIZE, 1,
HEADLINE_CLOSE_SIZE + 2, HEADLINE_CLOSE_SIZE + 2};
gdk_window_invalidate_rect(widget->window, &rect, TRUE);
}
return FALSE;
}
static void
reset_headline(GaimGtkBuddyList *gtkblist)
{
gtkblist->headline_callback = NULL;
gtkblist->headline_data = NULL;
gtkblist->headline_destroy = NULL;
gaim_gtk_set_urgent(GTK_WINDOW(gtkblist->window), FALSE);
}
static gboolean
headline_click_callback(gpointer data)
{
((GSourceFunc)gtkblist->headline_callback)(gtkblist->headline_data);
reset_headline(gtkblist);
return FALSE;
}
static gboolean
headline_box_press_cb(GtkWidget *widget, GdkEventButton *event, GaimGtkBuddyList *gtkblist)
{
gtk_widget_hide(gtkblist->headline_hbox);
if (gtkblist->headline_callback && !headline_hover_close((int)event->x, (int)event->y))
g_idle_add((GSourceFunc)headline_click_callback, gtkblist->headline_data);
else {
if (gtkblist->headline_destroy)
gtkblist->headline_destroy(gtkblist->headline_data);
reset_headline(gtkblist);
}
return TRUE;
}
/***********************************/
/* Connection error handling stuff */
/***********************************/
static void
ce_modify_account_cb(GaimAccount *account)
{
gaim_gtk_account_dialog_show(GAIM_GTK_MODIFY_ACCOUNT_DIALOG, account);
}
static void
ce_enable_account_cb(GaimAccount *account)
{
gaim_account_set_enabled(account, gaim_core_get_ui(), TRUE);
}
static void
connection_error_button_clicked_cb(GtkButton *widget, gpointer user_data)
{
GaimAccount *account;
char *primary;
const char *text;
gboolean enabled;
account = user_data;
primary = g_strdup_printf(_("%s disconnected"),
gaim_account_get_username(account));
text = g_hash_table_lookup(gtkblist->connection_errors, account);
enabled = gaim_account_get_enabled(account, gaim_core_get_ui());
gaim_request_action(account, _("Connection Error"), primary, text, 2,
account, 3,
_("OK"), NULL,
_("Modify Account"), GAIM_CALLBACK(ce_modify_account_cb),
enabled ? _("Connect") : _("Re-enable Account"),
enabled ? GAIM_CALLBACK(gaim_account_connect) :
GAIM_CALLBACK(ce_enable_account_cb));
g_free(primary);
gtk_widget_destroy(GTK_WIDGET(widget));
g_hash_table_remove(gtkblist->connection_errors, account);
}
/* Add some buttons that show connection errors */
static void
create_connection_error_buttons(gpointer key, gpointer value,
gpointer user_data)
{
GaimAccount *account;
GaimStatusType *status_type;
gchar *escaped, *text;
GtkWidget *button, *label, *image, *hbox;
GdkPixbuf *pixbuf;
account = key;
escaped = g_markup_escape_text((const gchar *)value, -1);
text = g_strdup_printf(_("<span color=\"red\">%s disconnected: %s</span>"),
gaim_account_get_username(account),
escaped);
g_free(escaped);
hbox = gtk_hbox_new(FALSE, 0);
/* Create the icon */
if ((status_type = gaim_account_get_status_type_with_primitive(account,
GAIM_STATUS_OFFLINE))) {
pixbuf = gaim_gtk_create_prpl_icon_with_status(account, status_type, 0.5);
if (pixbuf != NULL) {
image = gtk_image_new_from_pixbuf(pixbuf);
g_object_unref(pixbuf);
gtk_box_pack_start(GTK_BOX(hbox), image, FALSE, FALSE,
GAIM_HIG_BOX_SPACE);
}
}
/* Create the text */
label = gtk_label_new("");
gtk_label_set_markup(GTK_LABEL(label), text);
g_free(text);
#if GTK_CHECK_VERSION(2,6,0)
g_object_set(label, "ellipsize", PANGO_ELLIPSIZE_END, NULL);
#endif
gtk_box_pack_start(GTK_BOX(hbox), label, TRUE, TRUE,
GAIM_HIG_BOX_SPACE);
/* Create the actual button and put the icon and text on it */
button = gtk_button_new();
gtk_container_add(GTK_CONTAINER(button), hbox);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(connection_error_button_clicked_cb),
account);
gtk_widget_show_all(button);
gtk_box_pack_end(GTK_BOX(gtkblist->error_buttons), button,
FALSE, FALSE, 0);
}
void
gaim_gtk_blist_update_account_error_state(GaimAccount *account, const char *text)
{
GList *l;
if (text == NULL)
g_hash_table_remove(gtkblist->connection_errors, account);
else
g_hash_table_insert(gtkblist->connection_errors, account, g_strdup(text));
/* Remove the old error buttons */
for (l = gtk_container_get_children(GTK_CONTAINER(gtkblist->error_buttons));
l != NULL;
l = l->next)
{
gtk_widget_destroy(GTK_WIDGET(l->data));
}
/* Add new error buttons */
g_hash_table_foreach(gtkblist->connection_errors,
create_connection_error_buttons, NULL);
}
static gboolean
paint_headline_hbox (GtkWidget *widget,
GdkEventExpose *event,
gpointer user_data)
{
gtk_paint_flat_box (widget->style,
widget->window,
GTK_STATE_NORMAL,
GTK_SHADOW_OUT,
NULL,
widget,
"tooltip",
widget->allocation.x + 1,
widget->allocation.y + 1,
widget->allocation.width - 2,
widget->allocation.height - 2);
return FALSE;
}
static void
headline_style_set (GtkWidget *widget,
GtkStyle *prev_style)
{
GtkTooltips *tooltips;
GtkStyle *style;
if (gtkblist->changing_style)
return;
tooltips = gtk_tooltips_new ();
#if GLIB_CHECK_VERSION(2,10,0)
g_object_ref_sink (tooltips);
#else
g_object_ref(tooltips);
gtk_object_sink(GTK_OBJECT(tooltips));
#endif
gtk_tooltips_force_window (tooltips);
gtk_widget_ensure_style (tooltips->tip_window);
style = gtk_widget_get_style (tooltips->tip_window);
gtkblist->changing_style = TRUE;
gtk_widget_set_style (gtkblist->headline_hbox, style);
gtkblist->changing_style = FALSE;
g_object_unref (tooltips);
}
/******************************************/
/* End of connection error handling stuff */
/******************************************/
static int
blist_focus_cb(GtkWidget *widget, gpointer data, GaimGtkBuddyList *gtkblist)
{
gaim_gtk_set_urgent(GTK_WINDOW(gtkblist->window), FALSE);
return 0;
}
#if 0
static GtkWidget *
kiosk_page()
{
GtkWidget *ret = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
GtkWidget *label;
GtkWidget *entry;
GtkWidget *bbox;
GtkWidget *button;
label = gtk_label_new(NULL);
gtk_box_pack_start(GTK_BOX(ret), label, TRUE, TRUE, 0);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), _("<b>Username:</b>"));
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_box_pack_start(GTK_BOX(ret), label, FALSE, FALSE, 0);
entry = gtk_entry_new();
gtk_box_pack_start(GTK_BOX(ret), entry, FALSE, FALSE, 0);
label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), _("<b>Password:</b>"));
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_box_pack_start(GTK_BOX(ret), label, FALSE, FALSE, 0);
entry = gtk_entry_new();
gtk_entry_set_visibility(GTK_ENTRY(entry), FALSE);
gtk_box_pack_start(GTK_BOX(ret), entry, FALSE, FALSE, 0);
label = gtk_label_new(" ");
gtk_box_pack_start(GTK_BOX(ret), label, FALSE, FALSE, 0);
bbox = gtk_hbutton_box_new();
button = gtk_button_new_with_mnemonic(_("_Login"));
gtk_box_pack_start(GTK_BOX(ret), bbox, FALSE, FALSE, 0);
gtk_container_add(GTK_CONTAINER(bbox), button);
label = gtk_label_new(NULL);
gtk_box_pack_start(GTK_BOX(ret), label, TRUE, TRUE, 0);
gtk_container_set_border_width(GTK_CONTAINER(ret), GAIM_HIG_BORDER);
gtk_widget_show_all(ret);
return ret;
}
#endif
static void gaim_gtk_blist_show(GaimBuddyList *list)
{
void *handle;
GtkCellRenderer *rend;
GtkTreeViewColumn *column;
GtkWidget *menu;
GtkWidget *ebox;
GtkWidget *sw;
GtkWidget *sep;
GtkWidget *label;
GList *accounts;
char *pretty;
GtkAccelGroup *accel_group;
GtkTreeSelection *selection;
GtkTargetEntry dte[] = {{"GAIM_BLIST_NODE", GTK_TARGET_SAME_APP, DRAG_ROW},
{"application/x-im-contact", 0, DRAG_BUDDY},
{"text/x-vcard", 0, DRAG_VCARD },
{"text/uri-list", 0, DRAG_URI},
{"text/plain", 0, DRAG_TEXT}};
GtkTargetEntry ste[] = {{"GAIM_BLIST_NODE", GTK_TARGET_SAME_APP, DRAG_ROW},
{"application/x-im-contact", 0, DRAG_BUDDY},
{"text/x-vcard", 0, DRAG_VCARD }};
if (gtkblist && gtkblist->window) {
gaim_blist_set_visible(gaim_prefs_get_bool("/gaim/gtk/blist/list_visible"));
return;
}
gtkblist = GAIM_GTK_BLIST(list);
gtkblist->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_role(GTK_WINDOW(gtkblist->window), "buddy_list");
gtk_window_set_title(GTK_WINDOW(gtkblist->window), _("Buddy List"));
g_signal_connect(G_OBJECT(gtkblist->window), "focus-in-event",
G_CALLBACK(blist_focus_cb), gtkblist);
GTK_WINDOW(gtkblist->window)->allow_shrink = TRUE;
gtkblist->main_vbox = gtk_vbox_new(FALSE, 0);
gtk_widget_show(gtkblist->main_vbox);
gtk_container_add(GTK_CONTAINER(gtkblist->window), gtkblist->main_vbox);
g_signal_connect(G_OBJECT(gtkblist->window), "delete_event", G_CALLBACK(gtk_blist_delete_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->window), "configure_event", G_CALLBACK(gtk_blist_configure_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->window), "visibility_notify_event", G_CALLBACK(gtk_blist_visibility_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->window), "window_state_event", G_CALLBACK(gtk_blist_window_state_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->window), "key_press_event", G_CALLBACK(gtk_blist_window_key_press_cb), gtkblist);
gtk_widget_add_events(gtkblist->window, GDK_VISIBILITY_NOTIFY_MASK);
/******************************* Menu bar *************************************/
accel_group = gtk_accel_group_new();
gtk_window_add_accel_group(GTK_WINDOW (gtkblist->window), accel_group);
g_object_unref(accel_group);
gtkblist->ift = gtk_item_factory_new(GTK_TYPE_MENU_BAR, "<GaimMain>", accel_group);
gtk_item_factory_set_translate_func(gtkblist->ift,
(GtkTranslateFunc)item_factory_translate_func,
NULL, NULL);
gtk_item_factory_create_items(gtkblist->ift, sizeof(blist_menu) / sizeof(*blist_menu),
blist_menu, NULL);
gaim_gtk_load_accels();
g_signal_connect(G_OBJECT(accel_group), "accel-changed",
G_CALLBACK(gaim_gtk_save_accels_cb), NULL);
menu = gtk_item_factory_get_widget(gtkblist->ift, "<GaimMain>");
gtkblist->menutray = gaim_gtk_menu_tray_new();
gtk_menu_shell_append(GTK_MENU_SHELL(menu), gtkblist->menutray);
gtk_widget_show(gtkblist->menutray);
gtk_widget_show(menu);
gtk_box_pack_start(GTK_BOX(gtkblist->main_vbox), menu, FALSE, FALSE, 0);
accountmenu = gtk_item_factory_get_widget(gtkblist->ift, N_("/Accounts"));
/****************************** Notebook *************************************/
gtkblist->notebook = gtk_notebook_new();
gtk_notebook_set_show_tabs(GTK_NOTEBOOK(gtkblist->notebook), FALSE);
gtk_notebook_set_show_border(GTK_NOTEBOOK(gtkblist->notebook), FALSE);
gtk_box_pack_start(GTK_BOX(gtkblist->main_vbox), gtkblist->notebook, TRUE, TRUE, 0);
#if 0
gtk_notebook_append_page(GTK_NOTEBOOK(gtkblist->notebook), kiosk_page(), NULL);
#endif
/* Translators: Please maintain the use of -> and <- to refer to menu heirarchy */
pretty = gaim_gtk_make_pretty_arrows(_("<span weight='bold' size='larger'>Welcome to Gaim!</span>\n\n"
"You have no accounts enabled. Enable your IM accounts from the "
"<b>Accounts</b> window at <b>Accounts->Add/Edit</b>. Once you "
"enable accounts, you'll be able to sign on, set your status, "
"and talk to your friends."));
label = gtk_label_new(NULL);
gtk_widget_set_size_request(label, gaim_prefs_get_int("/gaim/gtk/blist/width") - 12, -1);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_misc_set_alignment(GTK_MISC(label), 0.5, 0.2);
gtk_label_set_markup(GTK_LABEL(label), pretty);
g_free(pretty);
gtk_notebook_append_page(GTK_NOTEBOOK(gtkblist->notebook),label, NULL);
gtkblist->vbox = gtk_vbox_new(FALSE, 0);
gtk_notebook_append_page(GTK_NOTEBOOK(gtkblist->notebook), gtkblist->vbox, NULL);
gtk_widget_show_all(gtkblist->notebook);
if ((accounts = gaim_accounts_get_all_active())) {
g_list_free(accounts);
gtk_notebook_set_current_page(GTK_NOTEBOOK(gtkblist->notebook), 1);
}
ebox = gtk_event_box_new();
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), ebox, FALSE, FALSE, 0);
gtkblist->headline_hbox = gtk_hbox_new(FALSE, 3);
gtk_container_set_border_width(GTK_CONTAINER(gtkblist->headline_hbox), 6);
gtk_container_add(GTK_CONTAINER(ebox), gtkblist->headline_hbox);
gtkblist->headline_image = gtk_image_new_from_pixbuf(NULL);
gtk_misc_set_alignment(GTK_MISC(gtkblist->headline_image), 0.0, 0);
gtkblist->headline_label = gtk_label_new(NULL);
gtk_widget_set_size_request(gtkblist->headline_label,
gaim_prefs_get_int("/gaim/gtk/blist/width")-25,-1);
gtk_label_set_line_wrap(GTK_LABEL(gtkblist->headline_label), TRUE);
gtk_box_pack_start(GTK_BOX(gtkblist->headline_hbox), gtkblist->headline_image, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(gtkblist->headline_hbox), gtkblist->headline_label, TRUE, TRUE, 0);
g_signal_connect(gtkblist->headline_hbox,
"style-set",
G_CALLBACK(headline_style_set),
NULL);
g_signal_connect (gtkblist->headline_hbox,
"expose_event",
G_CALLBACK (paint_headline_hbox),
NULL);
gtk_widget_set_name(gtkblist->headline_hbox, "gtk-tooltips");
gtkblist->headline_close = gtk_widget_render_icon(ebox, GTK_STOCK_CLOSE, -1, NULL);
if (gtkblist->headline_close) {
GdkPixbuf *scale = gdk_pixbuf_scale_simple(gtkblist->headline_close,
HEADLINE_CLOSE_SIZE, HEADLINE_CLOSE_SIZE, GDK_INTERP_BILINEAR);
gdk_pixbuf_unref(gtkblist->headline_close);
gtkblist->headline_close = scale;
}
gtkblist->hand_cursor = gdk_cursor_new (GDK_HAND2);
gtkblist->arrow_cursor = gdk_cursor_new (GDK_LEFT_PTR);
g_signal_connect(G_OBJECT(ebox), "enter-notify-event", G_CALLBACK(headline_box_enter_cb), gtkblist);
g_signal_connect(G_OBJECT(ebox), "leave-notify-event", G_CALLBACK(headline_box_leave_cb), gtkblist);
g_signal_connect(G_OBJECT(ebox), "button-press-event", G_CALLBACK(headline_box_press_cb), gtkblist);
#if 0
/* I couldn't get this to work. The idea was to draw the focus-border only
* when hovering over the close image. So for now, the focus-border is
* always there. -- sad */
gtk_widget_set_events(ebox, gtk_widget_get_events(ebox) | GDK_POINTER_MOTION_HINT_MASK);
g_signal_connect(G_OBJECT(ebox), "motion-notify-event", G_CALLBACK(headline_box_motion_cb), gtkblist);
#endif
/****************************** GtkTreeView **********************************/
sw = gtk_scrolled_window_new(NULL,NULL);
gtk_widget_show(sw);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_NONE);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtkblist->treemodel = gtk_tree_store_new(BLIST_COLUMNS,
GDK_TYPE_PIXBUF, /* Status icon */
G_TYPE_BOOLEAN, /* Status icon visible */
G_TYPE_STRING, /* Name */
G_TYPE_STRING, /* Idle */
G_TYPE_BOOLEAN, /* Idle visible */
GDK_TYPE_PIXBUF, /* Buddy icon */
G_TYPE_BOOLEAN, /* Buddy icon visible */
G_TYPE_POINTER, /* Node */
GDK_TYPE_COLOR, /* bgcolor */
G_TYPE_BOOLEAN, /* Group expander */
G_TYPE_BOOLEAN, /* Contact expander */
G_TYPE_BOOLEAN); /* Contact expander visible */
gtkblist->treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(gtkblist->treemodel));
gtk_widget_show(gtkblist->treeview);
gtk_widget_set_name(gtkblist->treeview, "gaim_gtkblist_treeview");
/* gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(gtkblist->treeview), TRUE); */
/* Set up selection stuff */
selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkblist->treeview));
g_signal_connect(G_OBJECT(selection), "changed", G_CALLBACK(gaim_gtk_blist_selection_changed), NULL);
/* Set up dnd */
gtk_tree_view_enable_model_drag_source(GTK_TREE_VIEW(gtkblist->treeview),
GDK_BUTTON1_MASK, ste, 3,
GDK_ACTION_COPY);
gtk_tree_view_enable_model_drag_dest(GTK_TREE_VIEW(gtkblist->treeview),
dte, 5,
GDK_ACTION_COPY | GDK_ACTION_MOVE);
g_signal_connect(G_OBJECT(gtkblist->treeview), "drag-data-received", G_CALLBACK(gaim_gtk_blist_drag_data_rcv_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "drag-data-get", G_CALLBACK(gaim_gtk_blist_drag_data_get_cb), NULL);
#ifdef _WIN32
g_signal_connect(G_OBJECT(gtkblist->treeview), "drag-begin", G_CALLBACK(gaim_gtk_blist_drag_begin), NULL);
#endif
g_signal_connect(G_OBJECT(gtkblist->treeview), "drag-motion", G_CALLBACK(gaim_gtk_blist_drag_motion_cb), NULL);
/* Tooltips */
g_signal_connect(G_OBJECT(gtkblist->treeview), "motion-notify-event", G_CALLBACK(gaim_gtk_blist_motion_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "leave-notify-event", G_CALLBACK(gaim_gtk_blist_leave_cb), NULL);
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(gtkblist->treeview), FALSE);
column = gtk_tree_view_column_new();
gtk_tree_view_append_column(GTK_TREE_VIEW(gtkblist->treeview), column);
gtk_tree_view_column_set_visible(column, FALSE);
gtk_tree_view_set_expander_column(GTK_TREE_VIEW(gtkblist->treeview), column);
gtkblist->text_column = column = gtk_tree_view_column_new ();
rend = gaim_gtk_cell_renderer_expander_new();
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend,
"expander-visible", GROUP_EXPANDER_COLUMN,
#if GTK_CHECK_VERSION(2,6,0)
"sensitive", GROUP_EXPANDER_COLUMN,
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
NULL);
rend = gaim_gtk_cell_renderer_expander_new();
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend,
"expander-visible", CONTACT_EXPANDER_COLUMN,
#if GTK_CHECK_VERSION(2,6,0)
"sensitive", CONTACT_EXPANDER_COLUMN,
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
"visible", CONTACT_EXPANDER_VISIBLE_COLUMN,
NULL);
rend = gtk_cell_renderer_pixbuf_new();
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend,
"pixbuf", STATUS_ICON_COLUMN,
"visible", STATUS_ICON_VISIBLE_COLUMN,
#if GTK_CHECK_VERSION(2,6,0)
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
NULL);
g_object_set(rend, "xalign", 0.0, "ypad", 0, NULL);
gtkblist->text_rend = rend = gtk_cell_renderer_text_new();
gtk_tree_view_column_pack_start (column, rend, TRUE);
gtk_tree_view_column_set_attributes(column, rend,
#if GTK_CHECK_VERSION(2,6,0)
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
"markup", NAME_COLUMN,
NULL);
g_signal_connect(G_OBJECT(rend), "edited", G_CALLBACK(gtk_blist_renderer_edited_cb), NULL);
g_object_set(rend, "ypad", 0, "yalign", 0.5, NULL);
#if GTK_CHECK_VERSION(2,6,0)
g_object_set(rend, "ellipsize", PANGO_ELLIPSIZE_END, NULL);
#endif
gtk_tree_view_append_column(GTK_TREE_VIEW(gtkblist->treeview), column);
rend = gtk_cell_renderer_text_new();
g_object_set(rend, "xalign", 1.0, "ypad", 0, NULL);
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend,
"markup", IDLE_COLUMN,
"visible", IDLE_VISIBLE_COLUMN,
#if GTK_CHECK_VERSION(2,6,0)
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
NULL);
rend = gtk_cell_renderer_pixbuf_new();
g_object_set(rend, "xalign", 1.0, "ypad", 0, NULL);
gtk_tree_view_column_pack_start(column, rend, FALSE);
gtk_tree_view_column_set_attributes(column, rend, "pixbuf", BUDDY_ICON_COLUMN,
#if GTK_CHECK_VERSION(2,6,0)
"cell-background-gdk", BGCOLOR_COLUMN,
#endif
"visible", BUDDY_ICON_VISIBLE_COLUMN,
NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "row-activated", G_CALLBACK(gtk_blist_row_activated_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "row-expanded", G_CALLBACK(gtk_blist_row_expanded_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "row-collapsed", G_CALLBACK(gtk_blist_row_collapsed_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "button-press-event", G_CALLBACK(gtk_blist_button_press_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "key-press-event", G_CALLBACK(gtk_blist_key_press_cb), NULL);
g_signal_connect(G_OBJECT(gtkblist->treeview), "popup-menu", G_CALLBACK(gaim_gtk_blist_popup_menu_cb), NULL);
/* Enable CTRL+F searching */
gtk_tree_view_set_search_column(GTK_TREE_VIEW(gtkblist->treeview), NAME_COLUMN);
gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(gtkblist->treeview), gaim_gtk_tree_view_search_equal_func, NULL, NULL);
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), sw, TRUE, TRUE, 0);
gtk_container_add(GTK_CONTAINER(sw), gtkblist->treeview);
sep = gtk_hseparator_new();
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), sep, FALSE, FALSE, 0);
gtkblist->scrollbook = gtk_gaim_scroll_book_new();
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), gtkblist->scrollbook, FALSE, FALSE, 0);
/* Create an empty vbox used for showing connection errors */
gtkblist->error_buttons = gtk_vbox_new(FALSE, 0);
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), gtkblist->error_buttons, FALSE, FALSE, 0);
/* Add the statusbox */
gtkblist->statusbox = gtk_gaim_status_box_new();
gtk_box_pack_start(GTK_BOX(gtkblist->vbox), gtkblist->statusbox, FALSE, TRUE, 0);
gtk_widget_set_name(gtkblist->statusbox, "gaim_gtkblist_statusbox");
gtk_container_set_border_width(GTK_CONTAINER(gtkblist->statusbox), 3);
gtk_widget_show(gtkblist->statusbox);
/* set the Show Offline Buddies option. must be done
* after the treeview or faceprint gets mad. -Robot101
*/
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item (gtkblist->ift, N_("/Buddies/Show Offline Buddies"))),
gaim_prefs_get_bool("/gaim/gtk/blist/show_offline_buddies"));
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item (gtkblist->ift, N_("/Buddies/Show Empty Groups"))),
gaim_prefs_get_bool("/gaim/gtk/blist/show_empty_groups"));
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item (gtkblist->ift, N_("/Tools/Mute Sounds"))),
gaim_prefs_get_bool("/gaim/gtk/sound/mute"));
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item (gtkblist->ift, N_("/Buddies/Show Buddy Details"))),
gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"));
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtk_item_factory_get_item (gtkblist->ift, N_("/Buddies/Show Idle Times"))),
gaim_prefs_get_bool("/gaim/gtk/blist/show_idle_time"));
if(!strcmp(gaim_prefs_get_string("/gaim/gtk/sound/method"), "none"))
gtk_widget_set_sensitive(gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools/Mute Sounds")), FALSE);
/* Update some dynamic things */
update_menu_bar(gtkblist);
gaim_gtk_blist_update_plugin_actions();
gaim_gtk_blist_update_sort_methods();
/* OK... let's show this bad boy. */
gaim_gtk_blist_refresh(list);
gaim_gtk_blist_restore_position();
gtk_widget_show_all(GTK_WIDGET(gtkblist->vbox));
gtk_widget_realize(GTK_WIDGET(gtkblist->window));
gaim_blist_set_visible(gaim_prefs_get_bool("/gaim/gtk/blist/list_visible"));
/* start the refresh timer */
gtkblist->refresh_timer = g_timeout_add(30000, (GSourceFunc)gaim_gtk_blist_refresh_timer, list);
handle = gaim_gtk_blist_get_handle();
/* things that affect how buddies are displayed */
gaim_prefs_connect_callback(handle, "/gaim/gtk/blist/show_buddy_icons",
_prefs_change_redo_list, NULL);
gaim_prefs_connect_callback(handle, "/gaim/gtk/blist/show_idle_time",
_prefs_change_redo_list, NULL);
gaim_prefs_connect_callback(handle, "/gaim/gtk/blist/show_empty_groups",
_prefs_change_redo_list, NULL);
gaim_prefs_connect_callback(handle, "/gaim/gtk/blist/show_offline_buddies",
_prefs_change_redo_list, NULL);
/* sorting */
gaim_prefs_connect_callback(handle, "/gaim/gtk/blist/sort_type",
_prefs_change_sort_method, NULL);
/* menus */
gaim_prefs_connect_callback(handle, "/gaim/gtk/sound/mute",
gaim_gtk_blist_mute_pref_cb, NULL);
gaim_prefs_connect_callback(handle, "/gaim/gtk/sound/method",
gaim_gtk_blist_sound_method_pref_cb, NULL);
/* Setup some gaim signal handlers. */
gaim_signal_connect(gaim_accounts_get_handle(), "account-enabled",
gtkblist, GAIM_CALLBACK(account_modified), gtkblist);
gaim_signal_connect(gaim_accounts_get_handle(), "account-disabled",
gtkblist, GAIM_CALLBACK(account_modified), gtkblist);
gaim_signal_connect(gaim_accounts_get_handle(), "account-removed",
gtkblist, GAIM_CALLBACK(account_modified), gtkblist);
gaim_signal_connect(gaim_accounts_get_handle(), "account-status-changed",
gtkblist, GAIM_CALLBACK(account_status_changed), gtkblist);
gaim_signal_connect(gaim_gtk_account_get_handle(), "account-modified",
gtkblist, GAIM_CALLBACK(account_modified), gtkblist);
gaim_signal_connect(gaim_connections_get_handle(), "signed-on",
gtkblist, GAIM_CALLBACK(sign_on_off_cb), list);
gaim_signal_connect(gaim_connections_get_handle(), "signed-off",
gtkblist, GAIM_CALLBACK(sign_on_off_cb), list);
gaim_signal_connect(gaim_plugins_get_handle(), "plugin-load",
gtkblist, GAIM_CALLBACK(plugin_changed_cb), NULL);
gaim_signal_connect(gaim_plugins_get_handle(), "plugin-unload",
gtkblist, GAIM_CALLBACK(plugin_changed_cb), NULL);
gaim_signal_connect(gaim_conversations_get_handle(), "conversation-updated",
gtkblist, GAIM_CALLBACK(conversation_updated_cb),
gtkblist);
gaim_signal_connect(gaim_conversations_get_handle(), "deleting-conversation",
gtkblist, GAIM_CALLBACK(conversation_deleting_cb),
gtkblist);
// gtk_widget_hide(gtkblist->scrollbook);
gtk_widget_hide(gtkblist->headline_hbox);
/* emit our created signal */
gaim_signal_emit(handle, "gtkblist-created", list);
}
static void redo_buddy_list(GaimBuddyList *list, gboolean remove, gboolean rerender)
{
GaimBlistNode *node;
gtkblist = GAIM_GTK_BLIST(list);
if(!gtkblist || !gtkblist->treeview)
return;
node = list->root;
while (node)
{
/* This is only needed when we're reverting to a non-GTK+ sorted
* status. We shouldn't need to remove otherwise.
*/
if (remove && !GAIM_BLIST_NODE_IS_GROUP(node))
gaim_gtk_blist_hide_node(list, node, FALSE);
if (GAIM_BLIST_NODE_IS_BUDDY(node))
gaim_gtk_blist_update_buddy(list, node, rerender);
else if (GAIM_BLIST_NODE_IS_CHAT(node))
gaim_gtk_blist_update(list, node);
else if (GAIM_BLIST_NODE_IS_GROUP(node))
gaim_gtk_blist_update(list, node);
node = gaim_blist_node_next(node, FALSE);
}
/* There is no hash table if there is nothing in the buddy list to update */
if (status_icon_hash_table) {
g_hash_table_destroy(status_icon_hash_table);
status_icon_hash_table = NULL;
}
}
void gaim_gtk_blist_refresh(GaimBuddyList *list)
{
redo_buddy_list(list, FALSE, TRUE);
}
void
gaim_gtk_blist_update_refresh_timeout()
{
GaimBuddyList *blist;
GaimGtkBuddyList *gtkblist;
blist = gaim_get_blist();
gtkblist = GAIM_GTK_BLIST(gaim_get_blist());
gtkblist->refresh_timer = g_timeout_add(30000,(GSourceFunc)gaim_gtk_blist_refresh_timer, blist);
}
static gboolean get_iter_from_node(GaimBlistNode *node, GtkTreeIter *iter) {
struct _gaim_gtk_blist_node *gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
GtkTreePath *path;
if (!gtknode) {
return FALSE;
}
if (!gtkblist) {
gaim_debug_error("gtkblist", "get_iter_from_node was called, but we don't seem to have a blist\n");
return FALSE;
}
if (!gtknode->row)
return FALSE;
if ((path = gtk_tree_row_reference_get_path(gtknode->row)) == NULL)
return FALSE;
if (!gtk_tree_model_get_iter(GTK_TREE_MODEL(gtkblist->treemodel), iter, path)) {
gtk_tree_path_free(path);
return FALSE;
}
gtk_tree_path_free(path);
return TRUE;
}
static void gaim_gtk_blist_remove(GaimBuddyList *list, GaimBlistNode *node)
{
struct _gaim_gtk_blist_node *gtknode = node->ui_data;
gaim_request_close_with_handle(node);
gaim_gtk_blist_hide_node(list, node, TRUE);
if(node->parent)
gaim_gtk_blist_update(list, node->parent);
/* There's something I don't understand here - Ethan */
/* Ethan said that back in 2003, but this g_free has been left commented
* out ever since. I can't find any reason at all why this is bad and
* valgrind found several reasons why it's good. If this causes problems
* comment it out again. Stu */
/* Of course it still causes problems - this breaks dragging buddies into
* contacts, the dragged buddy mysteriously 'disappears'. Stu. */
/* I think it's fixed now. Stu. */
if(gtknode) {
if(gtknode->recent_signonoff_timer > 0)
gaim_timeout_remove(gtknode->recent_signonoff_timer);
g_free(node->ui_data);
node->ui_data = NULL;
}
}
static gboolean do_selection_changed(GaimBlistNode *new_selection)
{
GaimBlistNode *old_selection = NULL;
/* test for gtkblist because crazy timeout means we can be called after the blist is gone */
if (gtkblist && new_selection != gtkblist->selected_node) {
old_selection = gtkblist->selected_node;
gtkblist->selected_node = new_selection;
if(new_selection)
gaim_gtk_blist_update(NULL, new_selection);
if(old_selection)
gaim_gtk_blist_update(NULL, old_selection);
}
return FALSE;
}
static void gaim_gtk_blist_selection_changed(GtkTreeSelection *selection, gpointer data)
{
GaimBlistNode *new_selection = NULL;
GtkTreeIter iter;
if(gtk_tree_selection_get_selected(selection, NULL, &iter)){
gtk_tree_model_get(GTK_TREE_MODEL(gtkblist->treemodel), &iter,
NODE_COLUMN, &new_selection, -1);
}
/* we set this up as a timeout, otherwise the blist flickers */
g_timeout_add(0, (GSourceFunc)do_selection_changed, new_selection);
}
static gboolean insert_node(GaimBuddyList *list, GaimBlistNode *node, GtkTreeIter *iter)
{
GtkTreeIter parent_iter, cur, *curptr = NULL;
struct _gaim_gtk_blist_node *gtknode = node->ui_data;
GtkTreePath *newpath;
if(!iter)
return FALSE;
if(node->parent && !get_iter_from_node(node->parent, &parent_iter))
return FALSE;
if(get_iter_from_node(node, &cur))
curptr = &cur;
if(GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_CHAT(node)) {
current_sort_method->func(node, list, parent_iter, curptr, iter);
} else {
sort_method_none(node, list, parent_iter, curptr, iter);
}
if(gtknode != NULL) {
gtk_tree_row_reference_free(gtknode->row);
} else {
gaim_gtk_blist_new_node(node);
gtknode = (struct _gaim_gtk_blist_node *)node->ui_data;
}
newpath = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel),
iter);
gtknode->row =
gtk_tree_row_reference_new(GTK_TREE_MODEL(gtkblist->treemodel),
newpath);
gtk_tree_path_free(newpath);
gtk_tree_store_set(gtkblist->treemodel, iter,
NODE_COLUMN, node,
-1);
if(node->parent) {
GtkTreePath *expand = NULL;
struct _gaim_gtk_blist_node *gtkparentnode = node->parent->ui_data;
if(GAIM_BLIST_NODE_IS_GROUP(node->parent)) {
if(!gaim_blist_node_get_bool(node->parent, "collapsed"))
expand = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel), &parent_iter);
} else if(GAIM_BLIST_NODE_IS_CONTACT(node->parent) &&
gtkparentnode->contact_expanded) {
expand = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel), &parent_iter);
}
if(expand) {
gtk_tree_view_expand_row(GTK_TREE_VIEW(gtkblist->treeview), expand, FALSE);
gtk_tree_path_free(expand);
}
}
return TRUE;
}
/*This version of gaim_gtk_blist_update_group can take the original buddy
or a group, but has much better algorithmic performance with a pre-known buddy*/
static void gaim_gtk_blist_update_group(GaimBuddyList *list, GaimBlistNode *node)
{
GaimGroup *group;
int count;
gboolean show = FALSE;
GaimBlistNode* gnode;
g_return_if_fail(node != NULL);
if (GAIM_BLIST_NODE_IS_GROUP(node))
gnode = node;
else if (GAIM_BLIST_NODE_IS_BUDDY(node))
gnode = node->parent->parent;
else if (GAIM_BLIST_NODE_IS_CONTACT(node) || GAIM_BLIST_NODE_IS_CHAT(node))
gnode = node->parent;
else
return;
group = (GaimGroup*)gnode;
if(gaim_prefs_get_bool("/gaim/gtk/blist/show_offline_buddies"))
count = gaim_blist_get_group_size(group, FALSE);
else
count = gaim_blist_get_group_online_count(group);
if (count > 0 || gaim_prefs_get_bool("/gaim/gtk/blist/show_empty_groups"))
show = TRUE;
else if (GAIM_BLIST_NODE_IS_BUDDY(node)){ /* Or chat? */
if (buddy_is_displayable((GaimBuddy*)node))
show = TRUE;}
if (show) {
GtkTreeIter iter;
GtkTreePath *path;
gboolean expanded;
GdkColor bgcolor;
char *title;
if(!insert_node(list, gnode, &iter))
return;
bgcolor = gtkblist->treeview->style->bg[GTK_STATE_ACTIVE];
path = gtk_tree_model_get_path(GTK_TREE_MODEL(gtkblist->treemodel), &iter);
expanded = gtk_tree_view_row_expanded(GTK_TREE_VIEW(gtkblist->treeview), path);
gtk_tree_path_free(path);
title = gaim_get_group_title(gnode, expanded);
gtk_tree_store_set(gtkblist->treemodel, &iter,
STATUS_ICON_VISIBLE_COLUMN, FALSE,
STATUS_ICON_COLUMN, NULL,
NAME_COLUMN, title,
NODE_COLUMN, gnode,
BGCOLOR_COLUMN, &bgcolor,
GROUP_EXPANDER_COLUMN, TRUE,
CONTACT_EXPANDER_VISIBLE_COLUMN, FALSE,
BUDDY_ICON_VISIBLE_COLUMN, FALSE,
IDLE_VISIBLE_COLUMN, FALSE,
-1);
g_free(title);
} else {
gaim_gtk_blist_hide_node(list, gnode, TRUE);
}
}
static char *gaim_get_group_title(GaimBlistNode *gnode, gboolean expanded)
{
GaimGroup *group;
GdkColor textcolor;
gboolean selected;
char group_count[12] = "";
char *mark, *esc;
group = (GaimGroup*)gnode;
textcolor = gtkblist->treeview->style->fg[GTK_STATE_ACTIVE];
selected = gtkblist ? (gtkblist->selected_node == gnode) : FALSE;
if (!expanded) {
g_snprintf(group_count, sizeof(group_count), " (%d/%d)",
gaim_blist_get_group_online_count(group),
gaim_blist_get_group_size(group, FALSE));
}
esc = g_markup_escape_text(group->name, -1);
if (selected)
mark = g_strdup_printf("<span weight='bold'>%s</span>%s", esc, group_count);
else
mark = g_strdup_printf("<span color='#%02x%02x%02x' weight='bold'>%s</span>%s",
textcolor.red>>8, textcolor.green>>8, textcolor.blue>>8,
esc, group_count);
g_free(esc);
return mark;
}
static void buddy_node(GaimBuddy *buddy, GtkTreeIter *iter, GaimBlistNode *node)
{
GaimPresence *presence;
GdkPixbuf *status, *avatar;
char *mark;
char *idle = NULL;
gboolean expanded = ((struct _gaim_gtk_blist_node *)(node->parent->ui_data))->contact_expanded;
gboolean selected = (gtkblist->selected_node == node);
gboolean biglist = gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons");
presence = gaim_buddy_get_presence(buddy);
status = gaim_gtk_blist_get_status_icon((GaimBlistNode*)buddy,
biglist ? GAIM_STATUS_ICON_LARGE : GAIM_STATUS_ICON_SMALL);
avatar = gaim_gtk_blist_get_buddy_icon((GaimBlistNode *)buddy, TRUE, TRUE, TRUE);
mark = gaim_gtk_blist_get_name_markup(buddy, selected);
if (gaim_prefs_get_bool("/gaim/gtk/blist/show_idle_time") &&
gaim_presence_is_idle(presence) &&
!gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"))
{
time_t idle_secs = gaim_presence_get_idle_time(presence);
if (idle_secs > 0)
{
time_t t;
int ihrs, imin;
time(&t);
ihrs = (t - idle_secs) / 3600;
imin = ((t - idle_secs) / 60) % 60;
idle = g_strdup_printf("%d:%02d", ihrs, imin);
}
}
if (gaim_presence_is_idle(presence))
{
if (idle && !selected) {
char *i2 = g_strdup_printf("<span color='%s'>%s</span>",
dim_grey(), idle);
g_free(idle);
idle = i2;
}
}
gtk_tree_store_set(gtkblist->treemodel, iter,
STATUS_ICON_COLUMN, status,
STATUS_ICON_VISIBLE_COLUMN, TRUE,
NAME_COLUMN, mark,
IDLE_COLUMN, idle,
IDLE_VISIBLE_COLUMN, !biglist && idle,
BUDDY_ICON_COLUMN, avatar,
BUDDY_ICON_VISIBLE_COLUMN, biglist,
BGCOLOR_COLUMN, NULL,
CONTACT_EXPANDER_COLUMN, NULL,
CONTACT_EXPANDER_VISIBLE_COLUMN, expanded,
-1);
g_free(mark);
g_free(idle);
if(status)
g_object_unref(status);
if(avatar)
g_object_unref(avatar);
}
/* This is a variation on the original gtk_blist_update_contact. Here we
can know in advance which buddy has changed so we can just update that */
static void gaim_gtk_blist_update_contact(GaimBuddyList *list, GaimBlistNode *node)
{
GaimBlistNode *cnode;
GaimContact *contact;
GaimBuddy *buddy;
struct _gaim_gtk_blist_node *gtknode;
if (GAIM_BLIST_NODE_IS_BUDDY(node))
cnode = node->parent;
else
cnode = node;
g_return_if_fail(GAIM_BLIST_NODE_IS_CONTACT(cnode));
/* First things first, update the group */
if (GAIM_BLIST_NODE_IS_BUDDY(node))
gaim_gtk_blist_update_group(list, node);
else
gaim_gtk_blist_update_group(list, cnode->parent);
contact = (GaimContact*)cnode;
buddy = gaim_contact_get_priority_buddy(contact);
if (buddy_is_displayable(buddy))
{
GtkTreeIter iter;
if(!insert_node(list, cnode, &iter))
return;
gtknode = (struct _gaim_gtk_blist_node *)cnode->ui_data;
if(gtknode->contact_expanded) {
GdkPixbuf *status;
char *mark;
status = gaim_gtk_blist_get_status_icon(cnode,
(gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons") ?
GAIM_STATUS_ICON_LARGE : GAIM_STATUS_ICON_SMALL));
mark = g_markup_escape_text(gaim_contact_get_alias(contact), -1);
gtk_tree_store_set(gtkblist->treemodel, &iter,
STATUS_ICON_COLUMN, status,
STATUS_ICON_VISIBLE_COLUMN, TRUE,
NAME_COLUMN, mark,
IDLE_COLUMN, NULL,
IDLE_VISIBLE_COLUMN, FALSE,
BGCOLOR_COLUMN, NULL,
BUDDY_ICON_COLUMN, NULL,
CONTACT_EXPANDER_COLUMN, TRUE,
CONTACT_EXPANDER_VISIBLE_COLUMN, TRUE,
-1);
g_free(mark);
if(status)
g_object_unref(status);
} else {
buddy_node(buddy, &iter, cnode);
}
} else {
gaim_gtk_blist_hide_node(list, cnode, TRUE);
}
}
static void gaim_gtk_blist_update_buddy(GaimBuddyList *list, GaimBlistNode *node, gboolean statusChange)
{
GaimBuddy *buddy;
struct _gaim_gtk_blist_node *gtkparentnode;
struct _gaim_gtk_blist_node *gtknode = node->ui_data;
g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(node));
if (node->parent == NULL)
return;
buddy = (GaimBuddy*)node;
if (statusChange)
gaim_gtk_blist_update_buddy_status_icon_key(gtknode, buddy,
(gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons")
? GAIM_STATUS_ICON_LARGE : GAIM_STATUS_ICON_SMALL));
/* First things first, update the contact */
gaim_gtk_blist_update_contact(list, node);
gtkparentnode = (struct _gaim_gtk_blist_node *)node->parent->ui_data;
if (gtkparentnode->contact_expanded && buddy_is_displayable(buddy))
{
GtkTreeIter iter;
if (!insert_node(list, node, &iter))
return;
buddy_node(buddy, &iter, node);
} else {
gaim_gtk_blist_hide_node(list, node, TRUE);
}
}
static void gaim_gtk_blist_update_chat(GaimBuddyList *list, GaimBlistNode *node)
{
GaimChat *chat;
g_return_if_fail(GAIM_BLIST_NODE_IS_CHAT(node));
/* First things first, update the group */
gaim_gtk_blist_update_group(list, node->parent);
chat = (GaimChat*)node;
if(gaim_account_is_connected(chat->account)) {
GtkTreeIter iter;
GdkPixbuf *status;
char *mark;
if(!insert_node(list, node, &iter))
return;
status = gaim_gtk_blist_get_status_icon(node,
(gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons") ?
GAIM_STATUS_ICON_LARGE : GAIM_STATUS_ICON_SMALL));
mark = g_markup_escape_text(gaim_chat_get_name(chat), -1);
gtk_tree_store_set(gtkblist->treemodel, &iter,
STATUS_ICON_COLUMN, status,
STATUS_ICON_VISIBLE_COLUMN, TRUE,
NAME_COLUMN, mark,
-1);
g_free(mark);
if(status)
g_object_unref(status);
} else {
gaim_gtk_blist_hide_node(list, node, TRUE);
}
}
static void gaim_gtk_blist_update(GaimBuddyList *list, GaimBlistNode *node)
{
if (list)
gtkblist = GAIM_GTK_BLIST(list);
if(!gtkblist || !gtkblist->treeview || !node)
return;
if (node->ui_data == NULL)
gaim_gtk_blist_new_node(node);
switch(node->type) {
case GAIM_BLIST_GROUP_NODE:
gaim_gtk_blist_update_group(list, node);
break;
case GAIM_BLIST_CONTACT_NODE:
gaim_gtk_blist_update_contact(list, node);
break;
case GAIM_BLIST_BUDDY_NODE:
gaim_gtk_blist_update_buddy(list, node, TRUE);
break;
case GAIM_BLIST_CHAT_NODE:
gaim_gtk_blist_update_chat(list, node);
break;
case GAIM_BLIST_OTHER_NODE:
return;
}
#if !GTK_CHECK_VERSION(2,6,0)
gtk_tree_view_columns_autosize(GTK_TREE_VIEW(gtkblist->treeview));
#endif
}
static void gaim_gtk_blist_destroy(GaimBuddyList *list)
{
if (!gtkblist)
return;
gaim_signals_disconnect_by_handle(gtkblist);
if (gtkblist->headline_close)
gdk_pixbuf_unref(gtkblist->headline_close);
gtk_widget_destroy(gtkblist->window);
gaim_gtk_blist_tooltip_destroy();
if (gtkblist->refresh_timer)
g_source_remove(gtkblist->refresh_timer);
if (gtkblist->timeout)
g_source_remove(gtkblist->timeout);
if (gtkblist->drag_timeout)
g_source_remove(gtkblist->drag_timeout);
g_hash_table_destroy(gtkblist->connection_errors);
gtkblist->refresh_timer = 0;
gtkblist->timeout = 0;
gtkblist->drag_timeout = 0;
gtkblist->window = gtkblist->vbox = gtkblist->treeview = NULL;
gtkblist->treemodel = NULL;
g_object_unref(G_OBJECT(gtkblist->ift));
gdk_cursor_unref(gtkblist->hand_cursor);
gdk_cursor_unref(gtkblist->arrow_cursor);
gtkblist->hand_cursor = NULL;
gtkblist->arrow_cursor = NULL;
g_free(gtkblist);
accountmenu = NULL;
gtkblist = NULL;
gaim_prefs_disconnect_by_handle(gaim_gtk_blist_get_handle());
}
static void gaim_gtk_blist_set_visible(GaimBuddyList *list, gboolean show)
{
if (!(gtkblist && gtkblist->window))
return;
if (show) {
if(!GAIM_WINDOW_ICONIFIED(gtkblist->window) && !GTK_WIDGET_VISIBLE(gtkblist->window))
gaim_signal_emit(gaim_gtk_blist_get_handle(), "gtkblist-unhiding", gtkblist);
gaim_gtk_blist_restore_position();
gtk_window_present(GTK_WINDOW(gtkblist->window));
} else {
if(visibility_manager_count) {
gaim_signal_emit(gaim_gtk_blist_get_handle(), "gtkblist-hiding", gtkblist);
gtk_widget_hide(gtkblist->window);
} else {
if (!GTK_WIDGET_VISIBLE(gtkblist->window))
gtk_widget_show(gtkblist->window);
gtk_window_iconify(GTK_WINDOW(gtkblist->window));
}
}
}
static GList *
groups_tree(void)
{
GList *tmp = NULL;
char *tmp2;
GaimGroup *g;
GaimBlistNode *gnode;
if (gaim_get_blist()->root == NULL)
{
tmp2 = g_strdup(_("Buddies"));
tmp = g_list_append(tmp, tmp2);
}
else
{
for (gnode = gaim_get_blist()->root;
gnode != NULL;
gnode = gnode->next)
{
if (GAIM_BLIST_NODE_IS_GROUP(gnode))
{
g = (GaimGroup *)gnode;
tmp2 = g->name;
tmp = g_list_append(tmp, tmp2);
}
}
}
return tmp;
}
static void
add_buddy_select_account_cb(GObject *w, GaimAccount *account,
GaimGtkAddBuddyData *data)
{
/* Save our account */
data->account = account;
}
static void
destroy_add_buddy_dialog_cb(GtkWidget *win, GaimGtkAddBuddyData *data)
{
g_free(data);
}
static void
add_buddy_cb(GtkWidget *w, int resp, GaimGtkAddBuddyData *data)
{
const char *grp, *who, *whoalias;
GaimGroup *g;
GaimBuddy *b;
GaimConversation *c;
GaimBuddyIcon *icon;
if (resp == GTK_RESPONSE_OK)
{
who = gtk_entry_get_text(GTK_ENTRY(data->entry));
grp = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(data->combo)->entry));
whoalias = gtk_entry_get_text(GTK_ENTRY(data->entry_for_alias));
if (*whoalias == '\0')
whoalias = NULL;
if ((g = gaim_find_group(grp)) == NULL)
{
g = gaim_group_new(grp);
gaim_blist_add_group(g, NULL);
}
b = gaim_buddy_new(data->account, who, whoalias);
gaim_blist_add_buddy(b, NULL, g, NULL);
gaim_account_add_buddy(data->account, b);
/*
* XXX
* It really seems like it would be better if the call to
* gaim_account_add_buddy() and gaim_conversation_update() were done in
* blist.c, possibly in the gaim_blist_add_buddy() function. Maybe
* gaim_account_add_buddy() should be renamed to
* gaim_blist_add_new_buddy() or something, and have it call
* gaim_blist_add_buddy() after it creates it. --Mark
*
* No that's not good. blist.c should only deal with adding nodes to the
* local list. We need a new, non-gtk file that calls both
* gaim_account_add_buddy and gaim_blist_add_buddy().
* Or something. --Mark
*/
c = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, who, data->account);
if (c != NULL) {
icon = gaim_conv_im_get_icon(GAIM_CONV_IM(c));
if (icon != NULL)
gaim_buddy_icon_update(icon);
}
}
gtk_widget_destroy(data->window);
}
static void
gaim_gtk_blist_request_add_buddy(GaimAccount *account, const char *username,
const char *group, const char *alias)
{
GtkWidget *table;
GtkWidget *label;
GtkWidget *hbox;
GtkWidget *vbox;
GtkWidget *img;
GaimGtkBuddyList *gtkblist;
GaimGtkAddBuddyData *data = g_new0(GaimGtkAddBuddyData, 1);
data->account =
(account != NULL
? account
: gaim_connection_get_account(gaim_connections_get_all()->data));
img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_QUESTION,
GTK_ICON_SIZE_DIALOG);
gtkblist = GAIM_GTK_BLIST(gaim_get_blist());
data->window = gtk_dialog_new_with_buttons(_("Add Buddy"),
NULL, GTK_DIALOG_NO_SEPARATOR,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_ADD, GTK_RESPONSE_OK,
NULL);
gtk_dialog_set_default_response(GTK_DIALOG(data->window), GTK_RESPONSE_OK);
gtk_container_set_border_width(GTK_CONTAINER(data->window), GAIM_HIG_BOX_SPACE);
gtk_window_set_resizable(GTK_WINDOW(data->window), FALSE);
gtk_box_set_spacing(GTK_BOX(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BORDER);
gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BOX_SPACE);
gtk_window_set_role(GTK_WINDOW(data->window), "add_buddy");
gtk_window_set_type_hint(GTK_WINDOW(data->window),
GDK_WINDOW_TYPE_HINT_DIALOG);
hbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
gtk_container_add(GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), hbox);
gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
vbox = gtk_vbox_new(FALSE, 0);
gtk_container_add(GTK_CONTAINER(hbox), vbox);
label = gtk_label_new(
_("Please enter the screen name of the person you would like "
"to add to your buddy list. You may optionally enter an alias, "
"or nickname, for the buddy. The alias will be displayed in "
"place of the screen name whenever possible.\n"));
gtk_widget_set_size_request(GTK_WIDGET(label), 400, -1);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
gtk_container_add(GTK_CONTAINER(vbox), hbox);
g_signal_connect(G_OBJECT(data->window), "destroy",
G_CALLBACK(destroy_add_buddy_dialog_cb), data);
table = gtk_table_new(4, 2, FALSE);
gtk_table_set_row_spacings(GTK_TABLE(table), 5);
gtk_table_set_col_spacings(GTK_TABLE(table), 5);
gtk_container_set_border_width(GTK_CONTAINER(table), 0);
gtk_box_pack_start(GTK_BOX(vbox), table, FALSE, FALSE, 0);
label = gtk_label_new(_("Screen name:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 0, 1);
data->entry = gtk_entry_new();
gtk_table_attach_defaults(GTK_TABLE(table), data->entry, 1, 2, 0, 1);
gtk_widget_grab_focus(data->entry);
if (username != NULL)
gtk_entry_set_text(GTK_ENTRY(data->entry), username);
else
gtk_dialog_set_response_sensitive(GTK_DIALOG(data->window),
GTK_RESPONSE_OK, FALSE);
gtk_entry_set_activates_default (GTK_ENTRY(data->entry), TRUE);
gaim_set_accessible_label (data->entry, label);
g_signal_connect(G_OBJECT(data->entry), "changed",
G_CALLBACK(gaim_gtk_set_sensitive_if_input),
data->window);
label = gtk_label_new(_("Alias:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 1, 2);
data->entry_for_alias = gtk_entry_new();
gtk_table_attach_defaults(GTK_TABLE(table),
data->entry_for_alias, 1, 2, 1, 2);
if (alias != NULL)
gtk_entry_set_text(GTK_ENTRY(data->entry_for_alias), alias);
if (username != NULL)
gtk_widget_grab_focus(GTK_WIDGET(data->entry_for_alias));
gtk_entry_set_activates_default (GTK_ENTRY(data->entry_for_alias), TRUE);
gaim_set_accessible_label (data->entry_for_alias, label);
label = gtk_label_new(_("Group:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 2, 3);
data->combo = gtk_combo_new();
gtk_combo_set_popdown_strings(GTK_COMBO(data->combo), groups_tree());
gtk_table_attach_defaults(GTK_TABLE(table), data->combo, 1, 2, 2, 3);
gaim_set_accessible_label (data->combo, label);
/* Set up stuff for the account box */
label = gtk_label_new(_("Account:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 3, 4);
data->account_box = gaim_gtk_account_option_menu_new(account, FALSE,
G_CALLBACK(add_buddy_select_account_cb), NULL, data);
gtk_table_attach_defaults(GTK_TABLE(table), data->account_box, 1, 2, 3, 4);
gaim_set_accessible_label (data->account_box, label);
/* End of account box */
g_signal_connect(G_OBJECT(data->window), "response",
G_CALLBACK(add_buddy_cb), data);
gtk_widget_show_all(data->window);
if (group != NULL)
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(data->combo)->entry), group);
}
static void
add_chat_cb(GtkWidget *w, GaimGtkAddChatData *data)
{
GHashTable *components;
GList *tmp;
GaimChat *chat;
GaimGroup *group;
const char *group_name;
const char *value;
components = g_hash_table_new_full(g_str_hash, g_str_equal,
g_free, g_free);
for (tmp = data->entries; tmp; tmp = tmp->next)
{
if (g_object_get_data(tmp->data, "is_spin"))
{
g_hash_table_replace(components,
g_strdup(g_object_get_data(tmp->data, "identifier")),
g_strdup_printf("%d",
gtk_spin_button_get_value_as_int(tmp->data)));
}
else
{
value = gtk_entry_get_text(tmp->data);
if (*value != '\0')
g_hash_table_replace(components,
g_strdup(g_object_get_data(tmp->data, "identifier")),
g_strdup(value));
}
}
chat = gaim_chat_new(data->account,
gtk_entry_get_text(GTK_ENTRY(data->alias_entry)),
components);
group_name = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(data->group_combo)->entry));
if ((group = gaim_find_group(group_name)) == NULL)
{
group = gaim_group_new(group_name);
gaim_blist_add_group(group, NULL);
}
if (chat != NULL)
{
gaim_blist_add_chat(chat, group, NULL);
}
gtk_widget_destroy(data->window);
g_free(data->default_chat_name);
g_list_free(data->entries);
g_free(data);
}
static void
add_chat_resp_cb(GtkWidget *w, int resp, GaimGtkAddChatData *data)
{
if (resp == GTK_RESPONSE_OK)
{
add_chat_cb(NULL, data);
}
else
{
gtk_widget_destroy(data->window);
g_free(data->default_chat_name);
g_list_free(data->entries);
g_free(data);
}
}
/*
* Check the values of all the text entry boxes. If any required input
* strings are empty then don't allow the user to click on "OK."
*/
static void
addchat_set_sensitive_if_input_cb(GtkWidget *entry, gpointer user_data)
{
GaimGtkAddChatData *data;
GList *tmp;
const char *text;
gboolean required;
gboolean sensitive = TRUE;
data = user_data;
for (tmp = data->entries; tmp != NULL; tmp = tmp->next)
{
if (!g_object_get_data(tmp->data, "is_spin"))
{
required = GPOINTER_TO_INT(g_object_get_data(tmp->data, "required"));
text = gtk_entry_get_text(tmp->data);
if (required && (*text == '\0'))
sensitive = FALSE;
}
}
gtk_dialog_set_response_sensitive(GTK_DIALOG(data->window), GTK_RESPONSE_OK, sensitive);
}
static void
rebuild_addchat_entries(GaimGtkAddChatData *data)
{
GaimConnection *gc;
GList *list = NULL, *tmp;
GHashTable *defaults = NULL;
struct proto_chat_entry *pce;
gboolean focus = TRUE;
g_return_if_fail(data->account != NULL);
gc = gaim_account_get_connection(data->account);
while ((tmp = gtk_container_get_children(GTK_CONTAINER(data->entries_box))))
gtk_widget_destroy(tmp->data);
g_list_free(data->entries);
data->entries = NULL;
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info != NULL)
list = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info(gc);
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info_defaults != NULL)
defaults = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->chat_info_defaults(gc, data->default_chat_name);
for (tmp = list; tmp; tmp = tmp->next)
{
GtkWidget *label;
GtkWidget *rowbox;
GtkWidget *input;
pce = tmp->data;
rowbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(data->entries_box), rowbox, FALSE, FALSE, 0);
label = gtk_label_new_with_mnemonic(pce->label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_size_group_add_widget(data->sg, label);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
if (pce->is_int)
{
GtkObject *adjust;
adjust = gtk_adjustment_new(pce->min, pce->min, pce->max,
1, 10, 10);
input = gtk_spin_button_new(GTK_ADJUSTMENT(adjust), 1, 0);
gtk_widget_set_size_request(input, 50, -1);
gtk_box_pack_end(GTK_BOX(rowbox), input, FALSE, FALSE, 0);
}
else
{
char *value;
input = gtk_entry_new();
gtk_entry_set_activates_default(GTK_ENTRY(input), TRUE);
value = g_hash_table_lookup(defaults, pce->identifier);
if (value != NULL)
gtk_entry_set_text(GTK_ENTRY(input), value);
if (pce->secret)
{
gtk_entry_set_visibility(GTK_ENTRY(input), FALSE);
if (gtk_entry_get_invisible_char(GTK_ENTRY(input)) == '*')
gtk_entry_set_invisible_char(GTK_ENTRY(input), GAIM_INVISIBLE_CHAR);
}
gtk_box_pack_end(GTK_BOX(rowbox), input, TRUE, TRUE, 0);
g_signal_connect(G_OBJECT(input), "changed",
G_CALLBACK(addchat_set_sensitive_if_input_cb), data);
}
/* Do the following for any type of input widget */
if (focus)
{
gtk_widget_grab_focus(input);
focus = FALSE;
}
gtk_label_set_mnemonic_widget(GTK_LABEL(label), input);
gaim_set_accessible_label(input, label);
g_object_set_data(G_OBJECT(input), "identifier", (gpointer)pce->identifier);
g_object_set_data(G_OBJECT(input), "is_spin", GINT_TO_POINTER(pce->is_int));
g_object_set_data(G_OBJECT(input), "required", GINT_TO_POINTER(pce->required));
data->entries = g_list_append(data->entries, input);
g_free(pce);
}
g_list_free(list);
g_hash_table_destroy(defaults);
/* Set whether the "OK" button should be clickable initially */
addchat_set_sensitive_if_input_cb(NULL, data);
gtk_widget_show_all(data->entries_box);
}
static void
addchat_select_account_cb(GObject *w, GaimAccount *account,
GaimGtkAddChatData *data)
{
if (strcmp(gaim_account_get_protocol_id(data->account),
gaim_account_get_protocol_id(account)) == 0)
{
data->account = account;
}
else
{
data->account = account;
rebuild_addchat_entries(data);
}
}
static void
gaim_gtk_blist_request_add_chat(GaimAccount *account, GaimGroup *group,
const char *alias, const char *name)
{
GaimGtkAddChatData *data;
GaimGtkBuddyList *gtkblist;
GList *l;
GaimConnection *gc;
GtkWidget *label;
GtkWidget *rowbox;
GtkWidget *hbox;
GtkWidget *vbox;
GtkWidget *img;
if (account != NULL) {
gc = gaim_account_get_connection(account);
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->join_chat == NULL) {
gaim_notify_error(gc, NULL, _("This protocol does not support chat rooms."), NULL);
return;
}
} else {
/* Find an account with chat capabilities */
for (l = gaim_connections_get_all(); l != NULL; l = l->next) {
gc = (GaimConnection *)l->data;
if (GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)->join_chat != NULL) {
account = gaim_connection_get_account(gc);
break;
}
}
if (account == NULL) {
gaim_notify_error(NULL, NULL,
_("You are not currently signed on with any "
"protocols that have the ability to chat."), NULL);
return;
}
}
data = g_new0(GaimGtkAddChatData, 1);
data->account = account;
data->default_chat_name = g_strdup(name);
img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_QUESTION,
GTK_ICON_SIZE_DIALOG);
gtkblist = GAIM_GTK_BLIST(gaim_get_blist());
data->sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
data->window = gtk_dialog_new_with_buttons(_("Add Chat"),
NULL, GTK_DIALOG_NO_SEPARATOR,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_ADD, GTK_RESPONSE_OK,
NULL);
gtk_dialog_set_default_response(GTK_DIALOG(data->window), GTK_RESPONSE_OK);
gtk_container_set_border_width(GTK_CONTAINER(data->window), GAIM_HIG_BOX_SPACE);
gtk_window_set_resizable(GTK_WINDOW(data->window), FALSE);
gtk_box_set_spacing(GTK_BOX(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BORDER);
gtk_container_set_border_width(GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), GAIM_HIG_BOX_SPACE);
gtk_window_set_role(GTK_WINDOW(data->window), "add_chat");
gtk_window_set_type_hint(GTK_WINDOW(data->window),
GDK_WINDOW_TYPE_HINT_DIALOG);
hbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
gtk_container_add(GTK_CONTAINER(GTK_DIALOG(data->window)->vbox), hbox);
gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
vbox = gtk_vbox_new(FALSE, 5);
gtk_container_add(GTK_CONTAINER(hbox), vbox);
label = gtk_label_new(
_("Please enter an alias, and the appropriate information "
"about the chat you would like to add to your buddy list.\n"));
gtk_widget_set_size_request(label, 400, -1);
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_box_pack_start(GTK_BOX(vbox), label, FALSE, FALSE, 0);
rowbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(vbox), rowbox, FALSE, FALSE, 0);
label = gtk_label_new(_("Account:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_size_group_add_widget(data->sg, label);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
data->account_menu = gaim_gtk_account_option_menu_new(account, FALSE,
G_CALLBACK(addchat_select_account_cb),
chat_account_filter_func, data);
gtk_box_pack_start(GTK_BOX(rowbox), data->account_menu, TRUE, TRUE, 0);
gaim_set_accessible_label (data->account_menu, label);
data->entries_box = gtk_vbox_new(FALSE, 5);
gtk_container_set_border_width(GTK_CONTAINER(data->entries_box), 0);
gtk_box_pack_start(GTK_BOX(vbox), data->entries_box, TRUE, TRUE, 0);
rebuild_addchat_entries(data);
rowbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(vbox), rowbox, FALSE, FALSE, 0);
label = gtk_label_new(_("Alias:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_size_group_add_widget(data->sg, label);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
data->alias_entry = gtk_entry_new();
if (alias != NULL)
gtk_entry_set_text(GTK_ENTRY(data->alias_entry), alias);
gtk_box_pack_end(GTK_BOX(rowbox), data->alias_entry, TRUE, TRUE, 0);
gtk_entry_set_activates_default(GTK_ENTRY(data->alias_entry), TRUE);
gaim_set_accessible_label (data->alias_entry, label);
rowbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(vbox), rowbox, FALSE, FALSE, 0);
label = gtk_label_new(_("Group:"));
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_size_group_add_widget(data->sg, label);
gtk_box_pack_start(GTK_BOX(rowbox), label, FALSE, FALSE, 0);
data->group_combo = gtk_combo_new();
gtk_combo_set_popdown_strings(GTK_COMBO(data->group_combo), groups_tree());
gtk_box_pack_end(GTK_BOX(rowbox), data->group_combo, TRUE, TRUE, 0);
if (group)
{
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(data->group_combo)->entry),
group->name);
}
gaim_set_accessible_label (data->group_combo, label);
g_signal_connect(G_OBJECT(data->window), "response",
G_CALLBACK(add_chat_resp_cb), data);
gtk_widget_show_all(data->window);
}
static void
add_group_cb(GaimConnection *gc, const char *group_name)
{
GaimGroup *group;
if ((group_name == NULL) || (*group_name == '\0'))
return;
group = gaim_group_new(group_name);
gaim_blist_add_group(group, NULL);
}
static void
gaim_gtk_blist_request_add_group(void)
{
gaim_request_input(NULL, _("Add Group"), NULL,
_("Please enter the name of the group to be added."),
NULL, FALSE, FALSE, NULL,
_("Add"), G_CALLBACK(add_group_cb),
_("Cancel"), NULL, NULL);
}
void
gaim_gtk_blist_toggle_visibility()
{
if (gtkblist && gtkblist->window) {
if (GTK_WIDGET_VISIBLE(gtkblist->window)) {
gaim_blist_set_visible(GAIM_WINDOW_ICONIFIED(gtkblist->window) || gtk_blist_obscured);
} else {
gaim_blist_set_visible(TRUE);
}
}
}
void
gaim_gtk_blist_visibility_manager_add()
{
visibility_manager_count++;
gaim_debug_info("gtkblist", "added visibility manager: %d\n", visibility_manager_count);
}
void
gaim_gtk_blist_visibility_manager_remove()
{
if (visibility_manager_count)
visibility_manager_count--;
if (!visibility_manager_count)
gaim_blist_set_visible(TRUE);
gaim_debug_info("gtkblist", "removed visibility manager: %d\n", visibility_manager_count);
}
void gaim_gtk_blist_add_alert(GtkWidget *widget)
{
gtk_container_add(GTK_CONTAINER(gtkblist->scrollbook), widget);
if (!GTK_WIDGET_HAS_FOCUS(gtkblist->window))
gaim_gtk_set_urgent(GTK_WINDOW(gtkblist->window), TRUE);
}
void
gaim_gtk_blist_set_headline(const char *text, GdkPixbuf *pixbuf, GCallback callback,
gpointer user_data, GDestroyNotify destroy)
{
/* Destroy any existing headline first */
if (gtkblist->headline_destroy)
gtkblist->headline_destroy(gtkblist->headline_data);
gtk_label_set_markup(GTK_LABEL(gtkblist->headline_label), text);
gtk_image_set_from_pixbuf(GTK_IMAGE(gtkblist->headline_image), pixbuf);
gtkblist->headline_callback = callback;
gtkblist->headline_data = user_data;
gtkblist->headline_destroy = destroy;
if (!GTK_WIDGET_HAS_FOCUS(gtkblist->window))
gaim_gtk_set_urgent(GTK_WINDOW(gtkblist->window), TRUE);
gtk_widget_show_all(gtkblist->headline_hbox);
}
static GaimBlistUiOps blist_ui_ops =
{
gaim_gtk_blist_new_list,
gaim_gtk_blist_new_node,
gaim_gtk_blist_show,
gaim_gtk_blist_update,
gaim_gtk_blist_remove,
gaim_gtk_blist_destroy,
gaim_gtk_blist_set_visible,
gaim_gtk_blist_request_add_buddy,
gaim_gtk_blist_request_add_chat,
gaim_gtk_blist_request_add_group
};
GaimBlistUiOps *
gaim_gtk_blist_get_ui_ops(void)
{
return &blist_ui_ops;
}
GaimGtkBuddyList *gaim_gtk_blist_get_default_gtk_blist()
{
return gtkblist;
}
static void account_signon_cb(GaimConnection *gc, gpointer z)
{
GaimAccount *account = gaim_connection_get_account(gc);
GaimBlistNode *gnode, *cnode;
for(gnode = gaim_get_blist()->root; gnode; gnode = gnode->next)
{
if(!GAIM_BLIST_NODE_IS_GROUP(gnode))
continue;
for(cnode = gnode->child; cnode; cnode = cnode->next)
{
GaimChat *chat;
if(!GAIM_BLIST_NODE_IS_CHAT(cnode))
continue;
chat = (GaimChat *)cnode;
if(chat->account != account)
continue;
if(gaim_blist_node_get_bool((GaimBlistNode*)chat, "gtk-autojoin") ||
(gaim_blist_node_get_string((GaimBlistNode*)chat,
"gtk-autojoin") != NULL))
serv_join_chat(gc, chat->components);
}
}
}
void *
gaim_gtk_blist_get_handle() {
static int handle;
return &handle;
}
static gboolean buddy_signonoff_timeout_cb(GaimBuddy *buddy)
{
struct _gaim_gtk_blist_node *gtknode = ((GaimBlistNode*)buddy)->ui_data;
gtknode->recent_signonoff = FALSE;
gtknode->recent_signonoff_timer = 0;
gaim_gtk_blist_update(NULL, (GaimBlistNode*)buddy);
return FALSE;
}
static void buddy_signonoff_cb(GaimBuddy *buddy)
{
struct _gaim_gtk_blist_node *gtknode;
if(!((GaimBlistNode*)buddy)->ui_data) {
gaim_gtk_blist_new_node((GaimBlistNode*)buddy);
}
gtknode = ((GaimBlistNode*)buddy)->ui_data;
gtknode->recent_signonoff = TRUE;
if(gtknode->recent_signonoff_timer > 0)
gaim_timeout_remove(gtknode->recent_signonoff_timer);
gtknode->recent_signonoff_timer = gaim_timeout_add(10000,
(GSourceFunc)buddy_signonoff_timeout_cb, buddy);
}
void gaim_gtk_blist_init(void)
{
void *gtk_blist_handle = gaim_gtk_blist_get_handle();
gaim_signal_connect(gaim_connections_get_handle(), "signed-on",
gtk_blist_handle, GAIM_CALLBACK(account_signon_cb),
NULL);
/* Initialize prefs */
gaim_prefs_add_none("/gaim/gtk/blist");
gaim_prefs_add_bool("/gaim/gtk/blist/show_buddy_icons", TRUE);
gaim_prefs_add_bool("/gaim/gtk/blist/show_empty_groups", FALSE);
gaim_prefs_add_bool("/gaim/gtk/blist/show_idle_time", TRUE);
gaim_prefs_add_bool("/gaim/gtk/blist/show_offline_buddies", FALSE);
gaim_prefs_add_bool("/gaim/gtk/blist/list_visible", TRUE);
gaim_prefs_add_bool("/gaim/gtk/blist/list_maximized", FALSE);
gaim_prefs_add_string("/gaim/gtk/blist/sort_type", "alphabetical");
gaim_prefs_add_int("/gaim/gtk/blist/x", 0);
gaim_prefs_add_int("/gaim/gtk/blist/y", 0);
gaim_prefs_add_int("/gaim/gtk/blist/width", 250); /* Golden ratio, baby */
gaim_prefs_add_int("/gaim/gtk/blist/height", 405); /* Golden ratio, baby */
gaim_prefs_add_int("/gaim/gtk/blist/tooltip_delay", 500);
/* Register our signals */
gaim_signal_register(gtk_blist_handle, "gtkblist-hiding",
gaim_marshal_VOID__POINTER, NULL, 1,
gaim_value_new(GAIM_TYPE_SUBTYPE,
GAIM_SUBTYPE_BLIST));
gaim_signal_register(gtk_blist_handle, "gtkblist-unhiding",
gaim_marshal_VOID__POINTER, NULL, 1,
gaim_value_new(GAIM_TYPE_SUBTYPE,
GAIM_SUBTYPE_BLIST));
gaim_signal_register(gtk_blist_handle, "gtkblist-created",
gaim_marshal_VOID__POINTER, NULL, 1,
gaim_value_new(GAIM_TYPE_SUBTYPE,
GAIM_SUBTYPE_BLIST));
gaim_signal_register(gtk_blist_handle, "drawing-tooltip",
gaim_marshal_VOID__POINTER_POINTER_UINT, NULL, 3,
gaim_value_new(GAIM_TYPE_SUBTYPE,
GAIM_SUBTYPE_BLIST_NODE),
gaim_value_new_outgoing(GAIM_TYPE_BOXED, "GString *"),
gaim_value_new(GAIM_TYPE_BOOLEAN));
gaim_signal_connect(gaim_blist_get_handle(), "buddy-signed-on", gtk_blist_handle, GAIM_CALLBACK(buddy_signonoff_cb), NULL);
gaim_signal_connect(gaim_blist_get_handle(), "buddy-signed-off", gtk_blist_handle, GAIM_CALLBACK(buddy_signonoff_cb), NULL);
gaim_signal_connect(gaim_blist_get_handle(), "buddy-privacy-changed", gtk_blist_handle, GAIM_CALLBACK(gaim_gtk_blist_update_privacy_cb), NULL);
}
void
gaim_gtk_blist_uninit(void) {
gaim_signals_unregister_by_instance(gaim_gtk_blist_get_handle());
gaim_signals_disconnect_by_handle(gaim_gtk_blist_get_handle());
}
/*********************************************************************
* Buddy List sorting functions *
*********************************************************************/
GList *gaim_gtk_blist_get_sort_methods()
{
return gaim_gtk_blist_sort_methods;
}
void gaim_gtk_blist_sort_method_reg(const char *id, const char *name, gaim_gtk_blist_sort_function func)
{
struct gaim_gtk_blist_sort_method *method = g_new0(struct gaim_gtk_blist_sort_method, 1);
method->id = g_strdup(id);
method->name = g_strdup(name);
method->func = func;
gaim_gtk_blist_sort_methods = g_list_append(gaim_gtk_blist_sort_methods, method);
gaim_gtk_blist_update_sort_methods();
}
void gaim_gtk_blist_sort_method_unreg(const char *id){
GList *l = gaim_gtk_blist_sort_methods;
while(l) {
struct gaim_gtk_blist_sort_method *method = l->data;
if(!strcmp(method->id, id)) {
gaim_gtk_blist_sort_methods = g_list_delete_link(gaim_gtk_blist_sort_methods, l);
g_free(method->id);
g_free(method->name);
g_free(method);
break;
}
}
gaim_gtk_blist_update_sort_methods();
}
void gaim_gtk_blist_sort_method_set(const char *id){
GList *l = gaim_gtk_blist_sort_methods;
if(!id)
id = "none";
while (l && strcmp(((struct gaim_gtk_blist_sort_method*)l->data)->id, id))
l = l->next;
if (l) {
current_sort_method = l->data;
} else if (!current_sort_method) {
gaim_gtk_blist_sort_method_set("none");
return;
}
if (!strcmp(id, "none")) {
redo_buddy_list(gaim_get_blist(), TRUE, FALSE);
} else {
redo_buddy_list(gaim_get_blist(), FALSE, FALSE);
}
}
/******************************************
** Sort Methods
******************************************/
static void sort_method_none(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter parent_iter, GtkTreeIter *cur, GtkTreeIter *iter)
{
GaimBlistNode *sibling = node->prev;
GtkTreeIter sibling_iter;
if (cur != NULL) {
*iter = *cur;
return;
}
while (sibling && !get_iter_from_node(sibling, &sibling_iter)) {
sibling = sibling->prev;
}
gtk_tree_store_insert_after(gtkblist->treemodel, iter,
node->parent ? &parent_iter : NULL,
sibling ? &sibling_iter : NULL);
}
#if GTK_CHECK_VERSION(2,2,1)
static void sort_method_alphabetical(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter)
{
GtkTreeIter more_z;
const char *my_name;
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
my_name = gaim_contact_get_alias((GaimContact*)node);
} else if(GAIM_BLIST_NODE_IS_CHAT(node)) {
my_name = gaim_chat_get_name((GaimChat*)node);
} else {
sort_method_none(node, blist, groupiter, cur, iter);
return;
}
if (!gtk_tree_model_iter_children(GTK_TREE_MODEL(gtkblist->treemodel), &more_z, &groupiter)) {
gtk_tree_store_insert(gtkblist->treemodel, iter, &groupiter, 0);
return;
}
do {
GValue val;
GaimBlistNode *n;
const char *this_name;
int cmp;
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &more_z, NODE_COLUMN, &val);
n = g_value_get_pointer(&val);
if(GAIM_BLIST_NODE_IS_CONTACT(n)) {
this_name = gaim_contact_get_alias((GaimContact*)n);
} else if(GAIM_BLIST_NODE_IS_CHAT(n)) {
this_name = gaim_chat_get_name((GaimChat*)n);
} else {
this_name = NULL;
}
cmp = gaim_utf8_strcasecmp(my_name, this_name);
if(this_name && (cmp < 0 || (cmp == 0 && node < n))) {
if(cur) {
gtk_tree_store_move_before(gtkblist->treemodel, cur, &more_z);
*iter = *cur;
return;
} else {
gtk_tree_store_insert_before(gtkblist->treemodel, iter,
&groupiter, &more_z);
return;
}
}
g_value_unset(&val);
} while (gtk_tree_model_iter_next (GTK_TREE_MODEL(gtkblist->treemodel), &more_z));
if(cur) {
gtk_tree_store_move_before(gtkblist->treemodel, cur, NULL);
*iter = *cur;
return;
} else {
gtk_tree_store_append(gtkblist->treemodel, iter, &groupiter);
return;
}
}
static void sort_method_status(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter)
{
GtkTreeIter more_z;
GaimBuddy *my_buddy, *this_buddy;
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
my_buddy = gaim_contact_get_priority_buddy((GaimContact*)node);
} else if(GAIM_BLIST_NODE_IS_CHAT(node)) {
if (cur != NULL) {
*iter = *cur;
return;
}
gtk_tree_store_append(gtkblist->treemodel, iter, &groupiter);
return;
} else {
sort_method_none(node, blist, groupiter, cur, iter);
return;
}
if (!gtk_tree_model_iter_children(GTK_TREE_MODEL(gtkblist->treemodel), &more_z, &groupiter)) {
gtk_tree_store_insert(gtkblist->treemodel, iter, &groupiter, 0);
return;
}
do {
GValue val;
GaimBlistNode *n;
gint name_cmp;
gint presence_cmp;
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &more_z, NODE_COLUMN, &val);
n = g_value_get_pointer(&val);
if(GAIM_BLIST_NODE_IS_CONTACT(n)) {
this_buddy = gaim_contact_get_priority_buddy((GaimContact*)n);
} else {
this_buddy = NULL;
}
name_cmp = gaim_utf8_strcasecmp(
gaim_contact_get_alias(gaim_buddy_get_contact(my_buddy)),
(this_buddy
? gaim_contact_get_alias(gaim_buddy_get_contact(this_buddy))
: NULL));
presence_cmp = gaim_presence_compare(
gaim_buddy_get_presence(my_buddy),
this_buddy ? gaim_buddy_get_presence(this_buddy) : NULL);
if (this_buddy == NULL ||
(presence_cmp < 0 ||
(presence_cmp == 0 &&
(name_cmp < 0 || (name_cmp == 0 && node < n)))))
{
if (cur != NULL)
{
gtk_tree_store_move_before(gtkblist->treemodel, cur, &more_z);
*iter = *cur;
return;
}
else
{
gtk_tree_store_insert_before(gtkblist->treemodel, iter,
&groupiter, &more_z);
return;
}
}
g_value_unset(&val);
}
while (gtk_tree_model_iter_next(GTK_TREE_MODEL(gtkblist->treemodel),
&more_z));
if (cur) {
gtk_tree_store_move_before(gtkblist->treemodel, cur, NULL);
*iter = *cur;
return;
} else {
gtk_tree_store_append(gtkblist->treemodel, iter, &groupiter);
return;
}
}
static void sort_method_log(GaimBlistNode *node, GaimBuddyList *blist, GtkTreeIter groupiter, GtkTreeIter *cur, GtkTreeIter *iter)
{
GtkTreeIter more_z;
int log_size = 0, this_log_size = 0;
const char *buddy_name, *this_buddy_name;
if(cur && (gtk_tree_model_iter_n_children(GTK_TREE_MODEL(gtkblist->treemodel), &groupiter) == 1)) {
*iter = *cur;
return;
}
if(GAIM_BLIST_NODE_IS_CONTACT(node)) {
GaimBlistNode *n;
for (n = node->child; n; n = n->next)
log_size += gaim_log_get_total_size(GAIM_LOG_IM, ((GaimBuddy*)(n))->name, ((GaimBuddy*)(n))->account);
buddy_name = gaim_contact_get_alias((GaimContact*)node);
} else if(GAIM_BLIST_NODE_IS_CHAT(node)) {
/* we don't have a reliable way of getting the log filename
* from the chat info in the blist, yet */
if (cur != NULL) {
*iter = *cur;
return;
}
gtk_tree_store_append(gtkblist->treemodel, iter, &groupiter);
return;
} else {
sort_method_none(node, blist, groupiter, cur, iter);
return;
}
if (!gtk_tree_model_iter_children(GTK_TREE_MODEL(gtkblist->treemodel), &more_z, &groupiter)) {
gtk_tree_store_insert(gtkblist->treemodel, iter, &groupiter, 0);
return;
}
do {
GValue val;
GaimBlistNode *n;
GaimBlistNode *n2;
int cmp;
val.g_type = 0;
gtk_tree_model_get_value (GTK_TREE_MODEL(gtkblist->treemodel), &more_z, NODE_COLUMN, &val);
n = g_value_get_pointer(&val);
this_log_size = 0;
if(GAIM_BLIST_NODE_IS_CONTACT(n)) {
for (n2 = n->child; n2; n2 = n2->next)
this_log_size += gaim_log_get_total_size(GAIM_LOG_IM, ((GaimBuddy*)(n2))->name, ((GaimBuddy*)(n2))->account);
this_buddy_name = gaim_contact_get_alias((GaimContact*)n);
} else {
this_buddy_name = NULL;
}
cmp = gaim_utf8_strcasecmp(buddy_name, this_buddy_name);
if (!GAIM_BLIST_NODE_IS_CONTACT(n) || log_size > this_log_size ||
((log_size == this_log_size) &&
(cmp < 0 || (cmp == 0 && node < n)))) {
if (cur != NULL) {
gtk_tree_store_move_before(gtkblist->treemodel, cur, &more_z);
*iter = *cur;
return;
} else {
gtk_tree_store_insert_before(gtkblist->treemodel, iter,
&groupiter, &more_z);
return;
}
}
g_value_unset(&val);
} while (gtk_tree_model_iter_next (GTK_TREE_MODEL(gtkblist->treemodel), &more_z));
if (cur != NULL) {
gtk_tree_store_move_before(gtkblist->treemodel, cur, NULL);
*iter = *cur;
return;
} else {
gtk_tree_store_append(gtkblist->treemodel, iter, &groupiter);
return;
}
}
#endif
static void
plugin_act(GtkObject *obj, GaimPluginAction *pam)
{
if (pam && pam->callback)
pam->callback(pam);
}
static void
build_plugin_actions(GtkWidget *menu, GaimPlugin *plugin)
{
GtkWidget *menuitem;
GaimPluginAction *action = NULL;
GList *actions, *l;
actions = GAIM_PLUGIN_ACTIONS(plugin, NULL);
for (l = actions; l != NULL; l = l->next)
{
if (l->data)
{
action = (GaimPluginAction *) l->data;
action->plugin = plugin;
action->context = NULL;
menuitem = gtk_menu_item_new_with_label(action->label);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
g_signal_connect(G_OBJECT(menuitem), "activate",
G_CALLBACK(plugin_act), action);
g_object_set_data_full(G_OBJECT(menuitem), "plugin_action",
action,
(GDestroyNotify)gaim_plugin_action_free);
gtk_widget_show(menuitem);
}
else
gaim_separator(menu);
}
g_list_free(actions);
}
static void
modify_account_cb(GtkWidget *widget, gpointer data)
{
gaim_gtk_account_dialog_show(GAIM_GTK_MODIFY_ACCOUNT_DIALOG, data);
}
static void
enable_account_cb(GtkCheckMenuItem *widget, gpointer data)
{
GaimAccount *account = data;
const GaimSavedStatus *saved_status;
saved_status = gaim_savedstatus_get_current();
gaim_savedstatus_activate_for_account(saved_status, account);
gaim_account_set_enabled(account, GAIM_GTK_UI, TRUE);
}
static void
disable_account_cb(GtkCheckMenuItem *widget, gpointer data)
{
GaimAccount *account = data;
gaim_account_set_enabled(account, GAIM_GTK_UI, FALSE);
}
void
gaim_gtk_blist_update_accounts_menu(void)
{
GtkWidget *menuitem = NULL, *submenu = NULL;
GtkAccelGroup *accel_group = NULL;
GList *l = NULL, *accounts = NULL;
gboolean disabled_accounts = FALSE;
if (accountmenu == NULL)
return;
/* Clear the old Accounts menu */
for (l = gtk_container_get_children(GTK_CONTAINER(accountmenu)); l; l = l->next) {
menuitem = l->data;
if (menuitem != gtk_item_factory_get_widget(gtkblist->ift, N_("/Accounts/Add\\/Edit")))
gtk_widget_destroy(menuitem);
}
for (accounts = gaim_accounts_get_all(); accounts; accounts = accounts->next) {
char *buf = NULL;
char *accel_path_buf = NULL;
GtkWidget *image = NULL;
GaimConnection *gc = NULL;
GaimAccount *account = NULL;
GaimStatus *status = NULL;
GdkPixbuf *pixbuf = NULL;
account = accounts->data;
accel_group = gtk_menu_get_accel_group(GTK_MENU(accountmenu));
if(gaim_account_get_enabled(account, GAIM_GTK_UI)) {
buf = g_strconcat(gaim_account_get_username(account), " (",
gaim_account_get_protocol_name(account), ")", NULL);
menuitem = gtk_image_menu_item_new_with_label(buf);
accel_path_buf = g_strconcat(N_("<GaimMain>/Accounts/"), buf, NULL);
g_free(buf);
status = gaim_account_get_active_status(account);
pixbuf = gaim_gtk_create_prpl_icon_with_status(account, gaim_status_get_type(status), 0.5);
if (pixbuf != NULL)
{
if (!gaim_account_is_connected(account))
gdk_pixbuf_saturate_and_pixelate(pixbuf, pixbuf,
0.0, FALSE);
image = gtk_image_new_from_pixbuf(pixbuf);
g_object_unref(G_OBJECT(pixbuf));
gtk_widget_show(image);
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem), image);
}
gtk_menu_shell_append(GTK_MENU_SHELL(accountmenu), menuitem);
gtk_widget_show(menuitem);
submenu = gtk_menu_new();
gtk_menu_set_accel_group(GTK_MENU(submenu), accel_group);
gtk_menu_set_accel_path(GTK_MENU(submenu), accel_path_buf);
g_free(accel_path_buf);
gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), submenu);
gtk_widget_show(submenu);
menuitem = gtk_menu_item_new_with_mnemonic(_("_Edit Account"));
g_signal_connect(G_OBJECT(menuitem), "activate",
G_CALLBACK(modify_account_cb), account);
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
gtk_widget_show(menuitem);
gaim_separator(submenu);
gc = gaim_account_get_connection(account);
if (gc && GAIM_CONNECTION_IS_CONNECTED(gc)) {
GaimPlugin *plugin = NULL;
plugin = gc->prpl;
if (GAIM_PLUGIN_HAS_ACTIONS(plugin)) {
GList *l, *ll = NULL;
GaimPluginAction *action = NULL;
for (l = ll = GAIM_PLUGIN_ACTIONS(plugin, gc); l; l = l->next) {
if (l->data) {
action = (GaimPluginAction *)l->data;
action->plugin = plugin;
action->context = gc;
menuitem = gtk_menu_item_new_with_label(action->label);
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
g_signal_connect(G_OBJECT(menuitem), "activate",
G_CALLBACK(plugin_act), action);
g_object_set_data_full(G_OBJECT(menuitem), "plugin_action", action, (GDestroyNotify)gaim_plugin_action_free);
gtk_widget_show(menuitem);
} else
gaim_separator(submenu);
}
} else {
menuitem = gtk_menu_item_new_with_label(_("No actions available"));
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
gtk_widget_set_sensitive(menuitem, FALSE);
gtk_widget_show(menuitem);
}
} else {
menuitem = gtk_menu_item_new_with_label(_("No actions available"));
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
gtk_widget_set_sensitive(menuitem, FALSE);
gtk_widget_show(menuitem);
}
gaim_separator(submenu);
menuitem = gtk_menu_item_new_with_mnemonic(_("_Disable"));
g_signal_connect(G_OBJECT(menuitem), "activate",
G_CALLBACK(disable_account_cb), account);
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
gtk_widget_show(menuitem);
} else {
disabled_accounts = TRUE;
}
}
if(disabled_accounts) {
gaim_separator(accountmenu);
menuitem = gtk_menu_item_new_with_label(_("Enable Account"));
gtk_menu_shell_append(GTK_MENU_SHELL(accountmenu), menuitem);
gtk_widget_show(menuitem);
submenu = gtk_menu_new();
gtk_menu_set_accel_group(GTK_MENU(submenu), accel_group);
gtk_menu_set_accel_path(GTK_MENU(submenu), N_("<GaimMain>/Accounts/Enable Account"));
gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), submenu);
gtk_widget_show(submenu);
for (accounts = gaim_accounts_get_all(); accounts; accounts = accounts->next) {
char *buf = NULL;
GtkWidget *image = NULL;
GaimAccount *account = NULL;
GdkPixbuf *pixbuf = NULL;
account = accounts->data;
if(!gaim_account_get_enabled(account, GAIM_GTK_UI)) {
disabled_accounts = TRUE;
buf = g_strconcat(gaim_account_get_username(account), " (",
gaim_account_get_protocol_name(account), ")", NULL);
menuitem = gtk_image_menu_item_new_with_label(buf);
g_free(buf);
pixbuf = gaim_gtk_create_prpl_icon(account, 0.5);
if (pixbuf != NULL)
{
if (!gaim_account_is_connected(account))
gdk_pixbuf_saturate_and_pixelate(pixbuf, pixbuf, 0.0, FALSE);
image = gtk_image_new_from_pixbuf(pixbuf);
g_object_unref(G_OBJECT(pixbuf));
gtk_widget_show(image);
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem), image);
}
g_signal_connect(G_OBJECT(menuitem), "activate",
G_CALLBACK(enable_account_cb), account);
gtk_menu_shell_append(GTK_MENU_SHELL(submenu), menuitem);
gtk_widget_show(menuitem);
}
}
}
}
static GList *plugin_submenus = NULL;
void
gaim_gtk_blist_update_plugin_actions(void)
{
GtkWidget *menuitem, *submenu;
GaimPlugin *plugin = NULL;
GList *l;
GtkAccelGroup *accel_group;
GtkWidget *pluginmenu = gtk_item_factory_get_widget(gtkblist->ift, N_("/Tools"));
g_return_if_fail(pluginmenu != NULL);
/* Remove old plugin action submenus from the Tools menu */
for (l = plugin_submenus; l; l = l->next)
gtk_widget_destroy(GTK_WIDGET(l->data));
g_list_free(plugin_submenus);
plugin_submenus = NULL;
accel_group = gtk_menu_get_accel_group(GTK_MENU(pluginmenu));
/* Add a submenu for each plugin with custom actions */
for (l = gaim_plugins_get_loaded(); l; l = l->next) {
char *path;
plugin = (GaimPlugin *) l->data;
if (GAIM_IS_PROTOCOL_PLUGIN(plugin))
continue;
if (!GAIM_PLUGIN_HAS_ACTIONS(plugin))
continue;
menuitem = gtk_image_menu_item_new_with_label(_(plugin->info->name));
gtk_menu_shell_append(GTK_MENU_SHELL(pluginmenu), menuitem);
gtk_widget_show(menuitem);
plugin_submenus = g_list_append(plugin_submenus, menuitem);
submenu = gtk_menu_new();
gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), submenu);
gtk_widget_show(submenu);
gtk_menu_set_accel_group(GTK_MENU(submenu), accel_group);
path = g_strdup_printf("%s/Tools/%s", gtkblist->ift->path, plugin->info->name);
gtk_menu_set_accel_path(GTK_MENU(submenu), path);
g_free(path);
build_plugin_actions(submenu, plugin);
}
}
static void
sortmethod_act(GtkCheckMenuItem *checkmenuitem, char *id)
{
if (gtk_check_menu_item_get_active(checkmenuitem))
{
gaim_gtk_set_cursor(gtkblist->window, GDK_WATCH);
/* This is redundant. I think. */
/* gaim_gtk_blist_sort_method_set(id); */
gaim_prefs_set_string("/gaim/gtk/blist/sort_type", id);
gaim_gtk_clear_cursor(gtkblist->window);
}
}
void
gaim_gtk_blist_update_sort_methods(void)
{
GtkWidget *menuitem = NULL, *activeitem = NULL;
GaimGtkBlistSortMethod *method = NULL;
GList *l;
GSList *sl = NULL;
GtkWidget *sortmenu;
const char *m = gaim_prefs_get_string("/gaim/gtk/blist/sort_type");
if ((gtkblist == NULL) || (gtkblist->ift == NULL))
return;
sortmenu = gtk_item_factory_get_widget(gtkblist->ift, N_("/Buddies/Sort Buddies"));
if (sortmenu == NULL)
return;
/* Clear the old menu */
for (l = gtk_container_get_children(GTK_CONTAINER(sortmenu)); l; l = l->next) {
menuitem = l->data;
gtk_widget_destroy(GTK_WIDGET(menuitem));
}
for (l = gaim_gtk_blist_sort_methods; l; l = l->next) {
method = (GaimGtkBlistSortMethod *) l->data;
menuitem = gtk_radio_menu_item_new_with_label(sl, _(method->name));
if (!strcmp(m, method->id))
activeitem = menuitem;
sl = gtk_radio_menu_item_get_group(GTK_RADIO_MENU_ITEM(menuitem));
gtk_menu_shell_append(GTK_MENU_SHELL(sortmenu), menuitem);
g_signal_connect(G_OBJECT(menuitem), "toggled",
G_CALLBACK(sortmethod_act), method->id);
gtk_widget_show(menuitem);
}
if (activeitem)
gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(activeitem), TRUE);
}