diff pidgin/gtkconv.c @ 15374:5fe8042783c1

Rename gtk/ and libgaim/ to pidgin/ and libpurple/
author Sean Egan <seanegan@gmail.com>
date Sat, 20 Jan 2007 02:32:10 +0000
parents
children c9497aad9fc4 29e443e0613f
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pidgin/gtkconv.c	Sat Jan 20 02:32:10 2007 +0000
@@ -0,0 +1,8540 @@
+/**
+ * @file gtkconv.c GTK+ Conversation 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"
+
+#ifndef _WIN32
+# include <X11/Xlib.h>
+#endif
+
+#ifdef USE_GTKSPELL
+# include <gtkspell/gtkspell.h>
+# ifdef _WIN32
+#  include "wspell.h"
+# endif
+#endif
+
+#include <gdk/gdkkeysyms.h>
+
+#include "account.h"
+#include "cmds.h"
+#include "debug.h"
+#include "idle.h"
+#include "imgstore.h"
+#include "log.h"
+#include "notify.h"
+#include "prpl.h"
+#include "request.h"
+#include "util.h"
+
+#include "gtkdnd-hints.h"
+#include "gtkblist.h"
+#include "gtkconv.h"
+#include "gtkconvwin.h"
+#include "gtkdialogs.h"
+#include "gtkimhtml.h"
+#include "gtkimhtmltoolbar.h"
+#include "gtklog.h"
+#include "gtkmenutray.h"
+#include "gtkpounce.h"
+#include "gtkprefs.h"
+#include "gtkprivacy.h"
+#include "gtkthemes.h"
+#include "gtkutils.h"
+#include "gaimstock.h"
+
+#include "gtknickcolors.h"
+
+#define AUTO_RESPONSE "&lt;AUTO-REPLY&gt; : "
+
+typedef  enum
+{
+	GAIM_GTKCONV_SET_TITLE 			= 1 << 0,
+	GAIM_GTKCONV_BUDDY_ICON			= 1 << 1,
+	GAIM_GTKCONV_MENU			= 1 << 2,
+	GAIM_GTKCONV_TAB_ICON			= 1 << 3,
+	GAIM_GTKCONV_TOPIC			= 1 << 4,
+	GAIM_GTKCONV_SMILEY_THEME		= 1 << 5,
+	GAIM_GTKCONV_COLORIZE_TITLE		= 1 << 6
+}GaimGtkConvFields;
+
+#define	GAIM_GTKCONV_ALL	((1 << 7) - 1)
+
+#define SEND_COLOR "#204a87"
+#define RECV_COLOR "#cc0000"
+#define HIGHLIGHT_COLOR "#AF7F00"
+
+/* Undef this to turn off "custom-smiley" debug messages */
+#define DEBUG_CUSTOM_SMILEY
+
+#define LUMINANCE(c) (float)((0.3*(c.red))+(0.59*(c.green))+(0.11*(c.blue)))
+
+#if 0
+/* These colors come from the default GNOME palette */
+static GdkColor nick_colors[] = {
+	{0, 47616, 46336, 43776},       /* Basic 3D Medium */
+	{0, 32768, 32000, 29696},       /* Basic 3D Dark */
+	{0, 22016, 20992, 18432},       /* 3D Shadow */
+	{0, 33536, 42496, 32512},       /* Green Medium */
+	{0, 23808, 29952, 21760},       /* Green Dark */
+	{0, 17408, 22016, 12800},       /* Green Shadow */
+	{0, 57344, 46592, 44800},       /* Red Hilight */
+	{0, 49408, 26112, 23040},       /* Red Medium */
+	{0, 34816, 17920, 12544},       /* Red Dark */
+	{0, 49408, 14336,  8704},       /* Red Shadow */
+	{0, 34816, 32512, 41728},       /* Purple Medium */
+	{0, 25088, 23296, 33024},       /* Purple Dark */
+	{0, 18688, 16384, 26112},       /* Purple Shadow */
+	{0, 40192, 47104, 53760},       /* Blue Hilight */
+	{0, 29952, 36864, 44544},       /* Blue Medium */
+	{0, 57344, 49920, 40448},       /* Face Skin Medium */
+	{0, 45824, 37120, 26880},       /* Face skin Dark */
+	{0, 33280, 26112, 18176},       /* Face Skin Shadow */
+	{0, 57088, 16896,  7680},       /* Accent Red */
+	{0, 39168,     0,     0},       /* Accent Red Dark */
+	{0, 17920, 40960, 17920},       /* Accent Green */
+	{0,  9728, 50944,  9728}        /* Accent Green Dark */
+};
+
+#define NUM_NICK_COLORS (sizeof(nick_colors) / sizeof(*nick_colors))
+#endif
+
+/* From http://www.w3.org/TR/AERT#color-contrast */
+#define MIN_BRIGHTNESS_CONTRAST 75
+#define MIN_COLOR_CONTRAST 200
+
+#define NUM_NICK_COLORS 220
+static GdkColor *nick_colors = NULL;
+static guint nbr_nick_colors;
+
+typedef struct {
+	GtkWidget *window;
+
+	GtkWidget *entry;
+	GtkWidget *message;
+
+	GaimConversation *conv;
+
+} InviteBuddyInfo;
+
+static GtkWidget *invite_dialog = NULL;
+static GtkWidget *warn_close_dialog = NULL;
+
+static GaimGtkWindow *hidden_convwin = NULL;
+static GList *window_list = NULL;
+
+
+static gboolean update_send_to_selection(GaimGtkWindow *win);
+static void generate_send_to_items(GaimGtkWindow *win);
+
+/* Prototypes. <-- because Paco-Paco hates this comment. */
+static void got_typing_keypress(GaimGtkConversation *gtkconv, gboolean first);
+static void gray_stuff_out(GaimGtkConversation *gtkconv);
+static GList *generate_invite_user_names(GaimConnection *gc);
+static void add_chat_buddy_common(GaimConversation *conv, GaimConvChatBuddy *cb, const char *old_name);
+static gboolean tab_complete(GaimConversation *conv);
+static void gaim_gtkconv_updated(GaimConversation *conv, GaimConvUpdateType type);
+static void gtkconv_set_unseen(GaimGtkConversation *gtkconv, GaimUnseenState state);
+static void update_typing_icon(GaimGtkConversation *gtkconv);
+static const char *item_factory_translate_func (const char *path, gpointer func_data);
+gboolean gaim_gtkconv_has_focus(GaimConversation *conv);
+static void gaim_gtkconv_custom_smiley_allocated(GdkPixbufLoader *loader, gpointer user_data);
+static void gaim_gtkconv_custom_smiley_closed(GdkPixbufLoader *loader, gpointer user_data);
+static GdkColor* generate_nick_colors(guint *numcolors, GdkColor background);
+static gboolean color_is_visible(GdkColor foreground, GdkColor background, int color_contrast, int brightness_contrast);
+static void gaim_gtkconv_update_fields(GaimConversation *conv, GaimGtkConvFields fields);
+static void focus_out_from_menubar(GtkWidget *wid, GaimGtkWindow *win);
+
+static GdkColor *get_nick_color(GaimGtkConversation *gtkconv, const char *name) {
+	static GdkColor col;
+	GtkStyle *style = gtk_widget_get_style(gtkconv->imhtml);
+	float scale;
+
+	col = nick_colors[g_str_hash(name) % nbr_nick_colors];
+	scale = ((1-(LUMINANCE(style->base[GTK_STATE_NORMAL]) / LUMINANCE(style->white))) *
+		       (LUMINANCE(style->white)/MAX(MAX(col.red, col.blue), col.green)));
+
+	/* The colors are chosen to look fine on white; we should never have to darken */
+	if (scale > 1) {
+		col.red   *= scale;
+		col.green *= scale;
+		col.blue  *= scale;
+	}
+
+	return &col;
+}
+
+/**************************************************************************
+ * Callbacks
+ **************************************************************************/
+
+static gint
+close_conv_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GList *list = g_list_copy(gtkconv->convs);
+
+	g_list_foreach(list, (GFunc)gaim_conversation_destroy, NULL);
+	g_list_free(list);
+
+	return TRUE;
+}
+
+static gboolean
+lbox_size_allocate_cb(GtkWidget *w, GtkAllocation *allocation, gpointer data)
+{
+	gaim_prefs_set_int("/gaim/gtk/conversations/chat/userlist_width", allocation->width == 1 ? 0 : allocation->width);
+
+	return FALSE;
+}
+
+static gboolean
+size_allocate_cb(GtkWidget *w, GtkAllocation *allocation, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+
+	if (!GTK_WIDGET_VISIBLE(w))
+		return FALSE;
+
+	if (!GAIM_IS_GTK_CONVERSATION(conv))
+		return FALSE;
+
+	if (gtkconv->auto_resize) {
+		return FALSE;
+	}
+
+	/* I find that I resize the window when it has a bunch of conversations in it, mostly so that the
+	 * tab bar will fit, but then I don't want new windows taking up the entire screen.  I check to see
+	 * if there is only one conversation in the window.  This way we'll be setting new windows to the
+	 * size of the last resized new window. */
+	/* I think that the above justification is not the majority, and that the new tab resizing should
+	 * negate it anyway.  --luke */
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+	{
+		if (w == gtkconv->imhtml) {
+			gaim_prefs_set_int("/gaim/gtk/conversations/im/default_width", allocation->width);
+			gaim_prefs_set_int("/gaim/gtk/conversations/im/default_height", allocation->height);
+		}
+		if (w == gtkconv->lower_hbox)
+			gaim_prefs_set_int("/gaim/gtk/conversations/im/entry_height", allocation->height);
+	}
+	else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+	{
+		if (w == gtkconv->imhtml) {
+			gaim_prefs_set_int("/gaim/gtk/conversations/chat/default_width", allocation->width);
+			gaim_prefs_set_int("/gaim/gtk/conversations/chat/default_height", allocation->height);
+		}
+		if (w == gtkconv->lower_hbox)
+			gaim_prefs_set_int("/gaim/gtk/conversations/chat/entry_height", allocation->height);
+	}
+
+	return FALSE;
+}
+
+static void
+default_formatize(GaimGtkConversation *c)
+{
+	GaimConversation *conv = c->active_conv;
+
+	if (conv->features & GAIM_CONNECTION_HTML)
+	{
+		char *color;
+		GdkColor fg_color, bg_color;
+
+		if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_bold") != GTK_IMHTML(c->entry)->edit.bold)
+			gtk_imhtml_toggle_bold(GTK_IMHTML(c->entry));
+
+		if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_italic") != GTK_IMHTML(c->entry)->edit.italic)
+			gtk_imhtml_toggle_italic(GTK_IMHTML(c->entry));
+
+		if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_underline") != GTK_IMHTML(c->entry)->edit.underline)
+			gtk_imhtml_toggle_underline(GTK_IMHTML(c->entry));
+
+		gtk_imhtml_toggle_fontface(GTK_IMHTML(c->entry),
+			gaim_prefs_get_string("/gaim/gtk/conversations/font_face"));
+
+		if (!(conv->features & GAIM_CONNECTION_NO_FONTSIZE))
+		{
+			int size = gaim_prefs_get_int("/gaim/gtk/conversations/font_size");
+
+			/* 3 is the default. */
+			if (size != 3)
+			{
+				gtk_imhtml_font_set_size(GTK_IMHTML(c->entry), size);
+			}
+		}
+
+		if(strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/fgcolor"), "") != 0)
+		{
+			gdk_color_parse(gaim_prefs_get_string("/gaim/gtk/conversations/fgcolor"),
+							&fg_color);
+			color = g_strdup_printf("#%02x%02x%02x",
+									fg_color.red   / 256,
+									fg_color.green / 256,
+									fg_color.blue  / 256);
+		}
+		else
+			color = g_strdup("");
+
+		gtk_imhtml_toggle_forecolor(GTK_IMHTML(c->entry), color);
+		g_free(color);
+
+		if(!(conv->features & GAIM_CONNECTION_NO_BGCOLOR) &&
+		   strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/bgcolor"), "") != 0)
+		{
+			gdk_color_parse(gaim_prefs_get_string("/gaim/gtk/conversations/bgcolor"),
+							&bg_color);
+			color = g_strdup_printf("#%02x%02x%02x",
+									bg_color.red   / 256,
+									bg_color.green / 256,
+									bg_color.blue  / 256);
+		}
+		else
+			color = g_strdup("");
+
+		gtk_imhtml_toggle_background(GTK_IMHTML(c->entry), color);
+		g_free(color);
+
+
+		if (conv->features & GAIM_CONNECTION_FORMATTING_WBFO)
+			gtk_imhtml_set_whole_buffer_formatting_only(GTK_IMHTML(c->entry), TRUE);
+		else
+			gtk_imhtml_set_whole_buffer_formatting_only(GTK_IMHTML(c->entry), FALSE);
+	}
+}
+
+static void
+clear_formatting_cb(GtkIMHtml *imhtml, GaimGtkConversation *gtkconv)
+{
+	default_formatize(gtkconv);
+}
+
+static const char *
+gaim_gtk_get_cmd_prefix(void)
+{
+	return "/";
+}
+
+static GaimCmdRet
+say_command_cb(GaimConversation *conv,
+              const char *cmd, char **args, char **error, void *data)
+{
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		gaim_conv_im_send(GAIM_CONV_IM(conv), args[0]);
+	else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+		gaim_conv_chat_send(GAIM_CONV_CHAT(conv), args[0]);
+
+	return GAIM_CMD_RET_OK;
+}
+
+static GaimCmdRet
+me_command_cb(GaimConversation *conv,
+              const char *cmd, char **args, char **error, void *data)
+{
+	char *tmp;
+
+	tmp = g_strdup_printf("/me %s", args[0]);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		gaim_conv_im_send(GAIM_CONV_IM(conv), tmp);
+	else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+		gaim_conv_chat_send(GAIM_CONV_CHAT(conv), tmp);
+
+	g_free(tmp);
+	return GAIM_CMD_RET_OK;
+}
+
+static GaimCmdRet
+debug_command_cb(GaimConversation *conv,
+                 const char *cmd, char **args, char **error, void *data)
+{
+	char *tmp, *markup;
+	GaimCmdStatus status;
+
+	if (!g_ascii_strcasecmp(args[0], "version")) {
+		tmp = g_strdup_printf("me is using Gaim v%s.", VERSION);
+		markup = g_markup_escape_text(tmp, -1);
+
+		status = gaim_cmd_do_command(conv, tmp, markup, error);
+
+		g_free(tmp);
+		g_free(markup);
+		return status;
+	} else {
+		gaim_conversation_write(conv, NULL, _("Supported debug options are:  version"),
+		                        GAIM_MESSAGE_NO_LOG|GAIM_MESSAGE_ERROR, time(NULL));
+		return GAIM_CMD_STATUS_OK;
+	}
+}
+
+static GaimCmdRet
+clear_command_cb(GaimConversation *conv,
+                 const char *cmd, char **args, char **error, void *data)
+{
+	GaimGtkConversation *gtkconv = NULL;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtk_imhtml_clear(GTK_IMHTML(gtkconv->imhtml));
+	return GAIM_CMD_STATUS_OK;
+}
+
+static GaimCmdRet
+help_command_cb(GaimConversation *conv,
+                 const char *cmd, char **args, char **error, void *data)
+{
+	GList *l, *text;
+	GString *s;
+
+	if (args[0] != NULL) {
+		s = g_string_new("");
+		text = gaim_cmd_help(conv, args[0]);
+
+		if (text) {
+			for (l = text; l; l = l->next)
+				if (l->next)
+					g_string_append_printf(s, "%s\n", (char *)l->data);
+				else
+					g_string_append_printf(s, "%s", (char *)l->data);
+		} else {
+			g_string_append(s, _("No such command (in this context)."));
+		}
+	} else {
+		s = g_string_new(_("Use \"/help &lt;command&gt;\" for help on a specific command.\n"
+											 "The following commands are available in this context:\n"));
+
+		text = gaim_cmd_list(conv);
+		for (l = text; l; l = l->next)
+			if (l->next)
+				g_string_append_printf(s, "%s, ", (char *)l->data);
+			else
+				g_string_append_printf(s, "%s.", (char *)l->data);
+		g_list_free(text);
+	}
+
+	gaim_conversation_write(conv, NULL, s->str, GAIM_MESSAGE_NO_LOG, time(NULL));
+	g_string_free(s, TRUE);
+
+	return GAIM_CMD_STATUS_OK;
+}
+
+static void
+send_history_add(GaimGtkConversation *gtkconv, const char *message)
+{
+	GList *first;
+
+	first = g_list_first(gtkconv->send_history);
+	g_free(first->data);
+	first->data = g_strdup(message);
+	gtkconv->send_history = g_list_prepend(first, NULL);
+}
+
+static void
+reset_default_size(GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+		gtk_widget_set_size_request(gtkconv->lower_hbox, -1,
+					    gaim_prefs_get_int("/gaim/gtk/conversations/chat/entry_height"));
+	else
+		gtk_widget_set_size_request(gtkconv->lower_hbox, -1,
+					    gaim_prefs_get_int("/gaim/gtk/conversations/im/entry_height"));
+}
+
+static gboolean
+check_for_and_do_command(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv;
+	char *cmd;
+	const char *prefix;
+	GtkTextIter start;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	prefix = gaim_gtk_get_cmd_prefix();
+
+	cmd = gtk_imhtml_get_text(GTK_IMHTML(gtkconv->entry), NULL, NULL);
+	gtk_text_buffer_get_start_iter(GTK_IMHTML(gtkconv->entry)->text_buffer, &start);
+
+	if (cmd && (strncmp(cmd, prefix, strlen(prefix)) == 0)
+	   && !gtk_text_iter_get_child_anchor(&start)) {
+		GaimCmdStatus status;
+		char *error, *cmdline, *markup, *send_history;
+		GtkTextIter end;
+
+		send_history = gtk_imhtml_get_markup(GTK_IMHTML(gtkconv->entry));
+		send_history_add(gtkconv, send_history);
+		g_free(send_history);
+
+		cmdline = cmd + strlen(prefix);
+
+		gtk_text_iter_forward_chars(&start, g_utf8_strlen(prefix, -1));
+		gtk_text_buffer_get_end_iter(GTK_IMHTML(gtkconv->entry)->text_buffer, &end);
+		markup = gtk_imhtml_get_markup_range(GTK_IMHTML(gtkconv->entry), &start, &end);
+		status = gaim_cmd_do_command(conv, cmdline, markup, &error);
+		g_free(cmd);
+		g_free(markup);
+
+		switch (status) {
+			case GAIM_CMD_STATUS_OK:
+				return TRUE;
+			case GAIM_CMD_STATUS_NOT_FOUND:
+				return FALSE;
+			case GAIM_CMD_STATUS_WRONG_ARGS:
+				gaim_conversation_write(conv, "", _("Syntax Error:  You typed the wrong number of arguments "
+								    "to that command."),
+						GAIM_MESSAGE_NO_LOG, time(NULL));
+				return TRUE;
+			case GAIM_CMD_STATUS_FAILED:
+				gaim_conversation_write(conv, "", error ? error : _("Your command failed for an unknown reason."),
+						GAIM_MESSAGE_NO_LOG, time(NULL));
+				g_free(error);
+				return TRUE;
+			case GAIM_CMD_STATUS_WRONG_TYPE:
+				if(gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+					gaim_conversation_write(conv, "", _("That command only works in chats, not IMs."),
+							GAIM_MESSAGE_NO_LOG, time(NULL));
+				else
+					gaim_conversation_write(conv, "", _("That command only works in IMs, not chats."),
+							GAIM_MESSAGE_NO_LOG, time(NULL));
+				return TRUE;
+			case GAIM_CMD_STATUS_WRONG_PRPL:
+				gaim_conversation_write(conv, "", _("That command doesn't work on this protocol."),
+						GAIM_MESSAGE_NO_LOG, time(NULL));
+				return TRUE;
+		}
+	}
+
+	g_free(cmd);
+	return FALSE;
+}
+
+static void
+send_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimAccount *account;
+	GaimConnection *gc;
+	GaimMessageFlags flags = 0;
+	char *buf, *clean;
+
+	account = gaim_conversation_get_account(conv);
+
+	if (!gaim_account_is_connected(account))
+		return;
+
+	if ((gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) &&
+		gaim_conv_chat_has_left(GAIM_CONV_CHAT(conv)))
+		return;
+
+	if (check_for_and_do_command(conv)) {
+		if (gtkconv->entry_growing) {
+			reset_default_size(gtkconv);
+			gtkconv->entry_growing = FALSE;
+		}
+		gtk_imhtml_clear(GTK_IMHTML(gtkconv->entry));
+		return;
+	}
+
+	buf = gtk_imhtml_get_markup(GTK_IMHTML(gtkconv->entry));
+	clean = gtk_imhtml_get_text(GTK_IMHTML(gtkconv->entry), NULL, NULL);
+
+	gtk_widget_grab_focus(gtkconv->entry);
+
+	if (strlen(clean) == 0) {
+		g_free(buf);
+		g_free(clean);
+		return;
+	}
+
+	gaim_idle_touch();
+
+	/* XXX: is there a better way to tell if the message has images? */
+	if (GTK_IMHTML(gtkconv->entry)->im_images != NULL)
+		flags |= GAIM_MESSAGE_IMAGES;
+
+	gc = gaim_account_get_connection(account);
+	if (gc && (conv->features & GAIM_CONNECTION_NO_NEWLINES)) {
+		char **bufs;
+		int i;
+
+		bufs = gtk_imhtml_get_markup_lines(GTK_IMHTML(gtkconv->entry));
+		for (i = 0; bufs[i]; i++) {
+			send_history_add(gtkconv, bufs[i]);
+			if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+				gaim_conv_im_send_with_flags(GAIM_CONV_IM(conv), bufs[i], flags);
+			else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+				gaim_conv_chat_send_with_flags(GAIM_CONV_CHAT(conv), bufs[i], flags);
+		}
+
+		g_strfreev(bufs);
+
+	} else {
+		send_history_add(gtkconv, buf);
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+			gaim_conv_im_send_with_flags(GAIM_CONV_IM(conv), buf, flags);
+		else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+			gaim_conv_chat_send_with_flags(GAIM_CONV_CHAT(conv), buf, flags);
+	}
+
+	g_free(clean);
+	g_free(buf);
+
+	gtk_imhtml_clear(GTK_IMHTML(gtkconv->entry));
+	if (gtkconv->entry_growing) {
+		reset_default_size(gtkconv);
+		gtkconv->entry_growing = FALSE;
+	}
+	gtkconv_set_unseen(gtkconv, GAIM_UNSEEN_NONE);
+}
+
+static void
+add_remove_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimAccount *account;
+	const char *name;
+	GaimConversation *conv = gtkconv->active_conv;
+
+	account = gaim_conversation_get_account(conv);
+	name    = gaim_conversation_get_name(conv);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		GaimBuddy *b;
+
+		b = gaim_find_buddy(account, name);
+		if (b != NULL)
+			gaim_gtkdialogs_remove_buddy(b);
+		else if (account != NULL && gaim_account_is_connected(account))
+			gaim_blist_request_add_buddy(account, (char *)name, NULL, NULL);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		GaimChat *c;
+
+		c = gaim_blist_find_chat(account, name);
+		if (c != NULL)
+			gaim_gtkdialogs_remove_chat(c);
+		else if (account != NULL && gaim_account_is_connected(account))
+			gaim_blist_request_add_chat(account, NULL, NULL, name);
+	}
+
+	gtk_widget_grab_focus(GAIM_GTK_CONVERSATION(conv)->entry);
+}
+
+static void chat_do_info(GaimGtkConversation *gtkconv, const char *who)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	GaimConnection *gc;
+
+	if ((gc = gaim_conversation_get_gc(conv))) {
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+		/*
+		 * If there are special needs for getting info on users in
+		 * buddy chat "rooms"...
+		 */
+		if (prpl_info->get_cb_info != NULL)
+		{
+			prpl_info->get_cb_info(gc,
+				gaim_conv_chat_get_id(GAIM_CONV_CHAT(conv)), who);
+		}
+		else
+			prpl_info->get_info(gc, who);
+	}
+}
+
+
+static void
+info_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		serv_get_info(gaim_conversation_get_gc(conv),
+					  gaim_conversation_get_name(conv));
+
+		gtk_widget_grab_focus(gtkconv->entry);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		/* Get info of the person currently selected in the GtkTreeView */
+		GaimGtkChatPane *gtkchat;
+		GtkTreeIter iter;
+		GtkTreeModel *model;
+		GtkTreeSelection *sel;
+		char *name;
+
+		gtkchat = gtkconv->u.chat;
+
+		model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+		sel   = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkchat->list));
+
+		if (gtk_tree_selection_get_selected(sel, NULL, &iter))
+			gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_NAME_COLUMN, &name, -1);
+		else
+			return;
+
+		chat_do_info(gtkconv, name);
+		g_free(name);
+	}
+}
+
+static void
+block_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimAccount *account;
+
+	account = gaim_conversation_get_account(conv);
+
+	if (account != NULL && gaim_account_is_connected(account))
+		gaim_gtk_request_add_block(account, gaim_conversation_get_name(conv));
+
+	gtk_widget_grab_focus(GAIM_GTK_CONVERSATION(conv)->entry);
+}
+
+static void
+do_invite(GtkWidget *w, int resp, InviteBuddyInfo *info)
+{
+	const char *buddy, *message;
+	GaimGtkConversation *gtkconv;
+
+	gtkconv = GAIM_GTK_CONVERSATION(info->conv);
+
+	if (resp == GTK_RESPONSE_OK) {
+		buddy   = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(info->entry)->entry));
+		message = gtk_entry_get_text(GTK_ENTRY(info->message));
+
+		if (!g_ascii_strcasecmp(buddy, ""))
+			return;
+
+		serv_chat_invite(gaim_conversation_get_gc(info->conv),
+						 gaim_conv_chat_get_id(GAIM_CONV_CHAT(info->conv)),
+						 message, buddy);
+	}
+
+	gtk_widget_destroy(invite_dialog);
+	invite_dialog = NULL;
+
+	g_free(info);
+}
+
+static void
+invite_dnd_recv(GtkWidget *widget, GdkDragContext *dc, gint x, gint y,
+				GtkSelectionData *sd, guint inf, guint t, gpointer data)
+{
+	InviteBuddyInfo *info = (InviteBuddyInfo *)data;
+	const char *convprotocol;
+
+	convprotocol = gaim_account_get_protocol_id(gaim_conversation_get_account(info->conv));
+
+	if (sd->target == gdk_atom_intern("GAIM_BLIST_NODE", FALSE))
+	{
+		GaimBlistNode *node = NULL;
+		GaimBuddy *buddy;
+
+		memcpy(&node, sd->data, sizeof(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;
+
+		if (strcmp(convprotocol, gaim_account_get_protocol_id(buddy->account)))
+		{
+			gaim_notify_error(GAIM_GTK_CONVERSATION(info->conv), NULL,
+							  _("That buddy is not on the same protocol as this "
+								"chat."), NULL);
+		}
+		else
+			gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(info->entry)->entry), buddy->name);
+
+		gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
+	}
+	else if (sd->target == gdk_atom_intern("application/x-im-contact", FALSE))
+	{
+		char *protocol = NULL;
+		char *username = NULL;
+		GaimAccount *account;
+
+		if (gaim_gtk_parse_x_im_contact((const char *)sd->data, FALSE, &account,
+										&protocol, &username, NULL))
+		{
+			if (account == NULL)
+			{
+				gaim_notify_error(GAIM_GTK_CONVERSATION(info->conv), NULL,
+					_("You are not currently signed on with an account that "
+					  "can invite that buddy."), NULL);
+			}
+			else if (strcmp(convprotocol, gaim_account_get_protocol_id(account)))
+			{
+				gaim_notify_error(GAIM_GTK_CONVERSATION(info->conv), NULL,
+								  _("That buddy is not on the same protocol as this "
+									"chat."), NULL);
+			}
+			else
+			{
+				gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(info->entry)->entry), username);
+			}
+		}
+
+		g_free(username);
+		g_free(protocol);
+
+		gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
+	}
+}
+
+static const GtkTargetEntry dnd_targets[] =
+{
+	{"GAIM_BLIST_NODE", GTK_TARGET_SAME_APP, 0},
+	{"application/x-im-contact", 0, 1}
+};
+
+static void
+invite_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	InviteBuddyInfo *info = NULL;
+
+	if (invite_dialog == NULL) {
+		GaimConnection *gc;
+		GaimGtkWindow *gtkwin;
+		GtkWidget *label;
+		GtkWidget *vbox, *hbox;
+		GtkWidget *table;
+		GtkWidget *img;
+
+		img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_QUESTION,
+		                               GTK_ICON_SIZE_DIALOG);
+
+		info = g_new0(InviteBuddyInfo, 1);
+		info->conv = conv;
+
+		gc        = gaim_conversation_get_gc(conv);
+		gtkwin    = gaim_gtkconv_get_window(gtkconv);
+
+		/* Create the new dialog. */
+		invite_dialog = gtk_dialog_new_with_buttons(
+			_("Invite Buddy Into Chat Room"),
+			GTK_WINDOW(gtkwin->window), 0,
+			GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
+			GAIM_STOCK_INVITE, GTK_RESPONSE_OK, NULL);
+
+		gtk_dialog_set_default_response(GTK_DIALOG(invite_dialog),
+		                                GTK_RESPONSE_OK);
+		gtk_container_set_border_width(GTK_CONTAINER(invite_dialog), GAIM_HIG_BOX_SPACE);
+		gtk_window_set_resizable(GTK_WINDOW(invite_dialog), FALSE);
+		gtk_dialog_set_has_separator(GTK_DIALOG(invite_dialog), FALSE);
+
+		info->window = GTK_WIDGET(invite_dialog);
+
+		/* Setup the outside spacing. */
+		vbox = GTK_DIALOG(invite_dialog)->vbox;
+
+		gtk_box_set_spacing(GTK_BOX(vbox), GAIM_HIG_BORDER);
+		gtk_container_set_border_width(GTK_CONTAINER(vbox), GAIM_HIG_BOX_SPACE);
+
+		/* Setup the inner hbox and put the dialog's icon in it. */
+		hbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
+		gtk_container_add(GTK_CONTAINER(vbox), hbox);
+		gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
+		gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
+
+		/* Setup the right vbox. */
+		vbox = gtk_vbox_new(FALSE, 0);
+		gtk_container_add(GTK_CONTAINER(hbox), vbox);
+
+		/* Put our happy label in it. */
+		label = gtk_label_new(_("Please enter the name of the user you wish "
+								"to invite, along with an optional invite "
+								"message."));
+		gtk_widget_set_size_request(label, 350, -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 for the table, and to give it some spacing on the left. */
+		hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+		gtk_container_add(GTK_CONTAINER(vbox), hbox);
+
+		/* Setup the table we're going to use to lay stuff out. */
+		table = gtk_table_new(2, 2, FALSE);
+		gtk_table_set_row_spacings(GTK_TABLE(table), GAIM_HIG_BOX_SPACE);
+		gtk_table_set_col_spacings(GTK_TABLE(table), GAIM_HIG_BOX_SPACE);
+		gtk_container_set_border_width(GTK_CONTAINER(table), GAIM_HIG_BORDER);
+		gtk_box_pack_start(GTK_BOX(vbox), table, FALSE, FALSE, 0);
+
+		/* Now the Buddy label */
+		label = gtk_label_new(NULL);
+		gtk_label_set_markup_with_mnemonic(GTK_LABEL(label), _("_Buddy:"));
+		gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
+		gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 0, 1);
+
+		/* Now the Buddy drop-down entry field. */
+		info->entry = gtk_combo_new();
+		gtk_combo_set_case_sensitive(GTK_COMBO(info->entry), FALSE);
+		gtk_entry_set_activates_default(
+				GTK_ENTRY(GTK_COMBO(info->entry)->entry), TRUE);
+
+		gtk_table_attach_defaults(GTK_TABLE(table), info->entry, 1, 2, 0, 1);
+		gtk_label_set_mnemonic_widget(GTK_LABEL(label), info->entry);
+
+		/* Fill in the names. */
+		gtk_combo_set_popdown_strings(GTK_COMBO(info->entry),
+									  generate_invite_user_names(gc));
+
+
+		/* Now the label for "Message" */
+		label = gtk_label_new(NULL);
+		gtk_label_set_markup_with_mnemonic(GTK_LABEL(label), _("_Message:"));
+		gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
+		gtk_table_attach_defaults(GTK_TABLE(table), label, 0, 1, 1, 2);
+
+
+		/* And finally, the Message entry field. */
+		info->message = gtk_entry_new();
+		gtk_entry_set_activates_default(GTK_ENTRY(info->message), TRUE);
+
+		gtk_table_attach_defaults(GTK_TABLE(table), info->message, 1, 2, 1, 2);
+		gtk_label_set_mnemonic_widget(GTK_LABEL(label), info->message);
+
+		/* Connect the signals. */
+		g_signal_connect(G_OBJECT(invite_dialog), "response",
+						 G_CALLBACK(do_invite), info);
+		/* Setup drag-and-drop */
+		gtk_drag_dest_set(info->window,
+						  GTK_DEST_DEFAULT_MOTION |
+						  GTK_DEST_DEFAULT_DROP,
+						  dnd_targets,
+						  sizeof(dnd_targets) / sizeof(GtkTargetEntry),
+						  GDK_ACTION_COPY);
+		gtk_drag_dest_set(info->entry,
+						  GTK_DEST_DEFAULT_MOTION |
+						  GTK_DEST_DEFAULT_DROP,
+						  dnd_targets,
+						  sizeof(dnd_targets) / sizeof(GtkTargetEntry),
+						  GDK_ACTION_COPY);
+
+		g_signal_connect(G_OBJECT(info->window), "drag_data_received",
+						 G_CALLBACK(invite_dnd_recv), info);
+		g_signal_connect(G_OBJECT(info->entry), "drag_data_received",
+						 G_CALLBACK(invite_dnd_recv), info);
+
+	}
+
+	gtk_widget_show_all(invite_dialog);
+
+	if (info != NULL)
+		gtk_widget_grab_focus(GTK_COMBO(info->entry)->entry);
+}
+
+static void
+menu_new_conv_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	gaim_gtkdialogs_im();
+}
+
+static void
+savelog_writefile_cb(void *user_data, const char *filename)
+{
+	GaimConversation *conv = (GaimConversation *)user_data;
+	FILE *fp;
+	const char *name;
+	gchar *text;
+
+	if ((fp = g_fopen(filename, "w+")) == NULL) {
+		gaim_notify_error(GAIM_GTK_CONVERSATION(conv), NULL, _("Unable to open file."), NULL);
+		return;
+	}
+
+	name = gaim_conversation_get_name(conv);
+	fprintf(fp, "<html>\n<head><title>%s</title></head>\n<body>", name);
+	fprintf(fp, _("<h1>Conversation with %s</h1>\n"), name);
+
+	text = gtk_imhtml_get_markup(
+		GTK_IMHTML(GAIM_GTK_CONVERSATION(conv)->imhtml));
+	fprintf(fp, "%s", text);
+	g_free(text);
+
+	fprintf(fp, "\n</body>\n</html>\n");
+	fclose(fp);
+}
+
+/*
+ * It would be kinda cool if this gave the option of saving a
+ * plaintext v. HTML file.
+ */
+static void
+menu_save_as_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv = gaim_gtk_conv_window_get_active_conversation(win);
+	gchar *buf;
+
+	buf = g_strdup_printf("%s.html", gaim_normalize(conv->account, conv->name));
+
+	gaim_request_file(GAIM_GTK_CONVERSATION(conv), _("Save Conversation"),
+					  gaim_escape_filename(buf),
+					  TRUE, G_CALLBACK(savelog_writefile_cb), NULL, conv);
+
+	g_free(buf);
+}
+
+static void
+menu_view_log_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimLogType type;
+	GaimGtkBuddyList *gtkblist;
+	GdkCursor *cursor;
+	const char *name;
+	GaimAccount *account;
+	GSList *buddies;
+	GSList *cur;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		type = GAIM_LOG_IM;
+	else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+		type = GAIM_LOG_CHAT;
+	else
+		return;
+
+	gtkblist = gaim_gtk_blist_get_default_gtk_blist();
+
+	cursor = gdk_cursor_new(GDK_WATCH);
+	gdk_window_set_cursor(gtkblist->window->window, cursor);
+	gdk_window_set_cursor(win->window->window, cursor);
+	gdk_cursor_unref(cursor);
+#if GTK_CHECK_VERSION(2,4,0)
+	gdk_display_flush(gdk_drawable_get_display(GDK_DRAWABLE(widget->window)));
+#else
+	gdk_flush();
+#endif
+
+	name = gaim_conversation_get_name(conv);
+	account = gaim_conversation_get_account(conv);
+
+	buddies = gaim_find_buddies(account, name);
+	for (cur = buddies; cur != NULL; cur = cur->next)
+	{
+		GaimBlistNode *node = cur->data;
+		if ((node != NULL) && ((node->prev != NULL) || (node->next != NULL)))
+		{
+			gaim_gtk_log_show_contact((GaimContact *)node->parent);
+			g_slist_free(buddies);
+			gdk_window_set_cursor(gtkblist->window->window, NULL);
+			gdk_window_set_cursor(win->window->window, NULL);
+			return;
+		}
+	}
+	g_slist_free(buddies);
+
+	gaim_gtk_log_show(type, name, account);
+
+	gdk_window_set_cursor(gtkblist->window->window, NULL);
+	gdk_window_set_cursor(win->window->window, NULL);
+}
+
+static void
+menu_clear_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtk_imhtml_clear(GTK_IMHTML(gtkconv->imhtml));
+}
+
+struct _search {
+	GaimGtkWindow *gtkwin;
+	GtkWidget *entry;
+};
+
+static void do_search_cb(GtkWidget *widget, gint resp, struct _search *s)
+{
+	GaimConversation *conv;
+	GaimGtkConversation *gtk_active_conv;
+	GList *iter;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(s->gtkwin);
+	gtk_active_conv = GAIM_GTK_CONVERSATION(conv);
+
+	switch (resp)
+	{
+		case GTK_RESPONSE_OK:
+			/* clear highlighting except the active conversation window
+			 * highlight the keywords in the active conversation window */
+			for (iter = gaim_gtk_conv_window_get_gtkconvs(s->gtkwin) ; iter ; iter = iter->next)
+			{
+				GaimGtkConversation *gtkconv = iter->data;
+
+				if (gtkconv != gtk_active_conv)
+				{
+					gtk_imhtml_search_clear(GTK_IMHTML(gtkconv->imhtml));
+				}
+				else
+				{
+					gtk_imhtml_search_find(GTK_IMHTML(gtk_active_conv->imhtml),
+					                       gtk_entry_get_text(GTK_ENTRY(s->entry)));
+				}
+			}
+			break;
+
+		case GTK_RESPONSE_DELETE_EVENT:
+		case GTK_RESPONSE_CLOSE:
+			/* clear the keyword highlighting in all the conversation windows */
+			for (iter = gaim_gtk_conv_window_get_gtkconvs(s->gtkwin); iter; iter=iter->next)
+			{
+				GaimGtkConversation *gconv = iter->data;
+				gtk_imhtml_search_clear(GTK_IMHTML(gconv->imhtml));
+			}
+
+			gtk_widget_destroy(s->gtkwin->dialogs.search);
+			s->gtkwin->dialogs.search = NULL;
+			g_free(s);
+			break;
+	}
+}
+
+static void
+menu_find_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *gtkwin = data;
+	GtkWidget *hbox;
+	GtkWidget *img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_QUESTION,
+											  GTK_ICON_SIZE_DIALOG);
+	GtkWidget *label;
+	struct _search *s;
+
+	if (gtkwin->dialogs.search) {
+		gtk_window_present(GTK_WINDOW(gtkwin->dialogs.search));
+		return;
+	}
+
+	s = g_malloc(sizeof(struct _search));
+	s->gtkwin = gtkwin;
+
+	gtkwin->dialogs.search = gtk_dialog_new_with_buttons(_("Find"),
+			GTK_WINDOW(gtkwin->window), GTK_DIALOG_DESTROY_WITH_PARENT,
+			GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
+			GTK_STOCK_FIND, GTK_RESPONSE_OK, NULL);
+	gtk_dialog_set_default_response(GTK_DIALOG(gtkwin->dialogs.search),
+									GTK_RESPONSE_OK);
+	g_signal_connect(G_OBJECT(gtkwin->dialogs.search), "response",
+					 G_CALLBACK(do_search_cb), s);
+
+	gtk_container_set_border_width(GTK_CONTAINER(gtkwin->dialogs.search), GAIM_HIG_BOX_SPACE);
+	gtk_window_set_resizable(GTK_WINDOW(gtkwin->dialogs.search), FALSE);
+	gtk_dialog_set_has_separator(GTK_DIALOG(gtkwin->dialogs.search), FALSE);
+	gtk_box_set_spacing(GTK_BOX(GTK_DIALOG(gtkwin->dialogs.search)->vbox), GAIM_HIG_BORDER);
+	gtk_container_set_border_width(
+		GTK_CONTAINER(GTK_DIALOG(gtkwin->dialogs.search)->vbox), GAIM_HIG_BOX_SPACE);
+
+	hbox = gtk_hbox_new(FALSE, GAIM_HIG_BORDER);
+	gtk_container_add(GTK_CONTAINER(GTK_DIALOG(gtkwin->dialogs.search)->vbox),
+					  hbox);
+	gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
+
+	gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
+	gtk_dialog_set_response_sensitive(GTK_DIALOG(gtkwin->dialogs.search),
+									  GTK_RESPONSE_OK, FALSE);
+
+	label = gtk_label_new(NULL);
+	gtk_label_set_markup_with_mnemonic(GTK_LABEL(label), _("_Search for:"));
+	gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+
+	s->entry = gtk_entry_new();
+	gtk_entry_set_activates_default(GTK_ENTRY(s->entry), TRUE);
+	gtk_label_set_mnemonic_widget(GTK_LABEL(label), GTK_WIDGET(s->entry));
+	g_signal_connect(G_OBJECT(s->entry), "changed",
+					 G_CALLBACK(gaim_gtk_set_sensitive_if_input),
+					 gtkwin->dialogs.search);
+	gtk_box_pack_start(GTK_BOX(hbox), s->entry, FALSE, FALSE, 0);
+
+	gtk_widget_show_all(gtkwin->dialogs.search);
+	gtk_widget_grab_focus(s->entry);
+}
+
+static void
+menu_send_file_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		serv_send_file(gaim_conversation_get_gc(conv), gaim_conversation_get_name(conv), NULL);
+	}
+
+}
+
+static void
+menu_add_pounce_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+
+	conv = gaim_gtk_conv_window_get_active_gtkconv(win)->active_conv;
+
+	gaim_gtk_pounce_editor_show(gaim_conversation_get_account(conv),
+								gaim_conversation_get_name(conv), NULL);
+}
+
+static void
+menu_insert_link_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimGtkConversation *gtkconv;
+	GtkIMHtmlToolbar *toolbar;
+
+	gtkconv    = gaim_gtk_conv_window_get_active_gtkconv(win);
+	toolbar = GTK_IMHTMLTOOLBAR(gtkconv->toolbar);
+
+	gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(toolbar->link),
+		!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(toolbar->link)));
+}
+
+static void
+menu_insert_image_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	GtkIMHtmlToolbar *toolbar;
+
+	gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+	conv    = gtkconv->active_conv;
+	toolbar = GTK_IMHTMLTOOLBAR(gtkconv->toolbar);
+
+	gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(toolbar->image),
+		!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(toolbar->image)));
+}
+
+static void
+menu_alias_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimAccount *account;
+	const char *name;
+
+	conv    = gaim_gtk_conv_window_get_active_conversation(win);
+	account = gaim_conversation_get_account(conv);
+	name    = gaim_conversation_get_name(conv);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		GaimBuddy *b;
+
+		b = gaim_find_buddy(account, name);
+		if (b != NULL)
+			gaim_gtkdialogs_alias_buddy(b);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		GaimChat *c;
+
+		c = gaim_blist_find_chat(account, name);
+		if (c != NULL)
+			gaim_gtkdialogs_alias_chat(c);
+	}
+}
+
+static void
+menu_get_info_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	info_cb(NULL, GAIM_GTK_CONVERSATION(conv));
+}
+
+static void
+menu_invite_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	invite_cb(NULL, GAIM_GTK_CONVERSATION(conv));
+}
+
+static void
+menu_block_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	block_cb(NULL, GAIM_GTK_CONVERSATION(conv));
+}
+
+static void
+menu_add_remove_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	add_remove_cb(NULL, GAIM_GTK_CONVERSATION(conv));
+}
+
+static void
+menu_close_conv_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+
+	close_conv_cb(NULL, GAIM_GTK_CONVERSATION(gaim_gtk_conv_window_get_active_conversation(win)));
+}
+
+static void
+menu_logging_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	gboolean logging;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (conv == NULL)
+		return;
+
+	logging = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
+
+	if (logging == gaim_conversation_is_logging(conv))
+		return;
+
+	if (logging)
+	{
+		/* Enable logging first so the message below can be logged. */
+		gaim_conversation_set_logging(conv, TRUE);
+
+		gaim_conversation_write(conv, NULL,
+								_("Logging started. Future messages in this conversation will be logged."),
+								conv->logs ? (GAIM_MESSAGE_SYSTEM) :
+								             (GAIM_MESSAGE_SYSTEM | GAIM_MESSAGE_NO_LOG),
+								time(NULL));
+	}
+	else
+	{
+		gaim_conversation_write(conv, NULL,
+								_("Logging stopped. Future messages in this conversation will not be logged."),
+								conv->logs ? (GAIM_MESSAGE_SYSTEM) :
+								             (GAIM_MESSAGE_SYSTEM | GAIM_MESSAGE_NO_LOG),
+								time(NULL));
+
+		/* Disable the logging second, so that the above message can be logged. */
+		gaim_conversation_set_logging(conv, FALSE);
+	}
+}
+
+static void
+menu_toolbar_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	gaim_prefs_set_bool("/gaim/gtk/conversations/show_formatting_toolbar",
+	                    gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget)));
+}
+
+static void
+menu_sounds_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (!conv)
+		return;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtkconv->make_sound =
+		gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
+}
+
+static void
+menu_timestamps_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	gaim_prefs_set_bool("/gaim/gtk/conversations/show_timestamps",
+		gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget)));
+}
+
+static void
+chat_do_im(GaimGtkConversation *gtkconv, const char *who)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimAccount *account;
+	GaimConnection *gc;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	char *real_who;
+
+	account = gaim_conversation_get_account(conv);
+	g_return_if_fail(account != NULL);
+
+	gc = gaim_account_get_connection(account);
+	g_return_if_fail(gc != NULL);
+
+	prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	if (prpl_info && prpl_info->get_cb_real_name)
+		real_who = prpl_info->get_cb_real_name(gc,
+				gaim_conv_chat_get_id(GAIM_CONV_CHAT(conv)), who);
+	else
+		real_who = g_strdup(who);
+
+	if(!real_who)
+		return;
+
+	gaim_gtkdialogs_im_with_user(account, real_who);
+
+	g_free(real_who);
+}
+
+static void
+ignore_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkChatPane *gtkchat;
+	GaimConvChatBuddy *cbuddy;
+	GaimConvChat *chat;
+	GaimConvChatBuddyFlags flags;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	GtkTreeSelection *sel;
+	char *name;
+	char *alias;
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkchat = gtkconv->u.chat;
+
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+	sel   = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkchat->list));
+
+	if (gtk_tree_selection_get_selected(sel, NULL, &iter)) {
+		gtk_tree_model_get(GTK_TREE_MODEL(model), &iter,
+						   CHAT_USERS_NAME_COLUMN, &name,
+						   CHAT_USERS_ALIAS_COLUMN, &alias,
+						   CHAT_USERS_FLAGS_COLUMN, &flags,
+						   -1);
+		gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
+	}
+	else
+		return;
+
+	if (gaim_conv_chat_is_user_ignored(chat, name))
+		gaim_conv_chat_unignore(chat, name);
+	else
+		gaim_conv_chat_ignore(chat, name);
+
+	cbuddy = gaim_conv_chat_cb_new(name, alias, flags);
+
+	add_chat_buddy_common(conv, cbuddy, NULL);
+	g_free(name);
+	g_free(alias);
+}
+
+static void
+menu_chat_im_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	const char *who = g_object_get_data(G_OBJECT(w), "user_data");
+
+	chat_do_im(gtkconv, who);
+}
+
+static void
+menu_chat_send_file_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	const char *who = g_object_get_data(G_OBJECT(w), "user_data");
+	GaimConnection *gc  = gaim_conversation_get_gc(conv);
+
+	serv_send_file(gc, who, NULL);
+}
+
+static void
+menu_chat_info_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	char *who;
+
+	who = g_object_get_data(G_OBJECT(w), "user_data");
+
+	chat_do_info(gtkconv, who);
+}
+
+static void
+menu_chat_get_away_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	GaimConnection *gc;
+	char *who;
+
+	gc  = gaim_conversation_get_gc(conv);
+	who = g_object_get_data(G_OBJECT(w), "user_data");
+
+	if (gc != NULL) {
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+		/*
+		 * May want to expand this to work similarly to menu_info_cb?
+		 */
+
+		if (prpl_info->get_cb_away != NULL)
+		{
+			prpl_info->get_cb_away(gc,
+				gaim_conv_chat_get_id(GAIM_CONV_CHAT(conv)), who);
+		}
+	}
+}
+
+static void
+menu_chat_add_remove_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimAccount *account;
+	GaimBuddy *b;
+	char *name;
+
+	account = gaim_conversation_get_account(conv);
+	name    = g_object_get_data(G_OBJECT(w), "user_data");
+	b       = gaim_find_buddy(account, name);
+
+	if (b != NULL)
+		gaim_gtkdialogs_remove_buddy(b);
+	else if (account != NULL && gaim_account_is_connected(account))
+		gaim_blist_request_add_buddy(account, name, NULL, NULL);
+
+	gtk_widget_grab_focus(GAIM_GTK_CONVERSATION(conv)->entry);
+}
+
+static GtkTextMark *
+get_mark_for_user(GaimGtkConversation *gtkconv, const char *who)
+{
+	GtkTextBuffer *buf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->imhtml));
+	char *tmp = g_strconcat("user:", who, NULL);
+	GtkTextMark *mark = gtk_text_buffer_get_mark(buf, tmp);
+
+	g_free(tmp);
+	return mark;
+}
+
+static void
+menu_last_said_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GtkTextMark *mark;
+	const char *who;
+
+	who = g_object_get_data(G_OBJECT(w), "user_data");
+	mark = get_mark_for_user(gtkconv, who);
+
+	if (mark != NULL)
+		gtk_text_view_scroll_to_mark(GTK_TEXT_VIEW(gtkconv->imhtml), mark, 0.1, FALSE, 0, 0);
+	else
+		g_return_if_reached();
+}
+
+static GtkWidget *
+create_chat_menu(GaimConversation *conv, const char *who, GaimConnection *gc)
+{
+	static GtkWidget *menu = NULL;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	GaimConvChat *chat = GAIM_CONV_CHAT(conv);
+	gboolean is_me = FALSE;
+	GtkWidget *button;
+	GaimBuddy *buddy = NULL;
+
+	if (gc != NULL)
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	/*
+	 * If a menu already exists, destroy it before creating a new one,
+	 * thus freeing-up the memory it occupied.
+	 */
+	if (menu)
+		gtk_widget_destroy(menu);
+
+	if (!strcmp(chat->nick, gaim_normalize(conv->account, who)))
+		is_me = TRUE;
+
+	menu = gtk_menu_new();
+
+	if (!is_me) {
+		button = gaim_new_item_from_stock(menu, _("IM"), GAIM_STOCK_IM,
+					G_CALLBACK(menu_chat_im_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+		if (gc == NULL)
+			gtk_widget_set_sensitive(button, FALSE);
+
+		g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+
+
+		if (prpl_info && prpl_info->send_file)
+		{
+			button = gaim_new_item_from_stock(menu, _("Send File"),
+				GAIM_STOCK_FILE_TRANSFER, G_CALLBACK(menu_chat_send_file_cb),
+				GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+			if (gc == NULL || prpl_info == NULL ||
+			    !(!prpl_info->can_receive_file || prpl_info->can_receive_file(gc, who)))
+			{
+				gtk_widget_set_sensitive(button, FALSE);
+			}
+
+			g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+		}
+
+
+		if (gaim_conv_chat_is_user_ignored(GAIM_CONV_CHAT(conv), who))
+			button = gaim_new_item_from_stock(menu, _("Un-Ignore"), GAIM_STOCK_IGNORE,
+							G_CALLBACK(ignore_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+		else
+			button = gaim_new_item_from_stock(menu, _("Ignore"), GAIM_STOCK_IGNORE,
+							G_CALLBACK(ignore_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+		if (gc == NULL)
+			gtk_widget_set_sensitive(button, FALSE);
+
+		g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+	}
+
+	if (prpl_info && (prpl_info->get_info || prpl_info->get_cb_info)) {
+		button = gaim_new_item_from_stock(menu, _("Info"), GAIM_STOCK_INFO,
+						G_CALLBACK(menu_chat_info_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+		if (gc == NULL)
+			gtk_widget_set_sensitive(button, FALSE);
+
+		g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+	}
+
+	if (prpl_info && prpl_info->get_cb_away) {
+		button = gaim_new_item_from_stock(menu, _("Get Away Message"), GAIM_STOCK_AWAY,
+					G_CALLBACK(menu_chat_get_away_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+		if (gc == NULL)
+			gtk_widget_set_sensitive(button, FALSE);
+
+		g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+	}
+
+	if (!is_me && prpl_info && !(prpl_info->options & OPT_PROTO_UNIQUE_CHATNAME)) {
+		if ((buddy = gaim_find_buddy(conv->account, who)) != NULL)
+			button = gaim_new_item_from_stock(menu, _("Remove"), GTK_STOCK_REMOVE,
+						G_CALLBACK(menu_chat_add_remove_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+		else
+			button = gaim_new_item_from_stock(menu, _("Add"), GTK_STOCK_ADD,
+						G_CALLBACK(menu_chat_add_remove_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+
+		if (gc == NULL)
+			gtk_widget_set_sensitive(button, FALSE);
+
+		g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+	}
+
+	button = gaim_new_item_from_stock(menu, _("Last said"), GTK_STOCK_INDEX,
+						G_CALLBACK(menu_last_said_cb), GAIM_GTK_CONVERSATION(conv), 0, 0, NULL);
+	g_object_set_data_full(G_OBJECT(button), "user_data", g_strdup(who), g_free);
+	if (!get_mark_for_user(GAIM_GTK_CONVERSATION(conv), who))
+		gtk_widget_set_sensitive(button, FALSE);
+
+	if (buddy != NULL)
+	{
+		if (gaim_account_is_connected(conv->account))
+			gaim_gtk_append_blist_node_proto_menu(menu, conv->account->gc,
+												  (GaimBlistNode *)buddy);
+		gaim_gtk_append_blist_node_extended_menu(menu, (GaimBlistNode *)buddy);
+		gtk_widget_show_all(menu);
+	}
+
+	return menu;
+}
+
+
+static gint
+gtkconv_chat_popup_menu_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkChatPane *gtkchat;
+	GaimConnection *gc;
+	GaimAccount *account;
+	GtkTreeSelection *sel;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	GtkWidget *menu;
+	gchar *who;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+	account = gaim_conversation_get_account(conv);
+	gc      = account->gc;
+
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+
+	sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkchat->list));
+	if(!gtk_tree_selection_get_selected(sel, NULL, &iter))
+		return FALSE;
+
+	gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_NAME_COLUMN, &who, -1);
+	menu = create_chat_menu (conv, who, gc);
+	gtk_menu_popup(GTK_MENU(menu), NULL, NULL,
+				   gaim_gtk_treeview_popup_menu_position_func, widget,
+				   0, GDK_CURRENT_TIME);
+	g_free(who);
+
+	return TRUE;
+}
+
+
+static gint
+right_click_chat_cb(GtkWidget *widget, GdkEventButton *event,
+					GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkChatPane *gtkchat;
+	GaimConnection *gc;
+	GaimAccount *account;
+	GtkTreePath *path;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	GtkTreeViewColumn *column;
+	gchar *who;
+	int x, y;
+
+	gtkchat = gtkconv->u.chat;
+	account = gaim_conversation_get_account(conv);
+	gc      = account->gc;
+
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+
+	gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(gtkchat->list),
+								  event->x, event->y, &path, &column, &x, &y);
+
+	if (path == NULL)
+		return FALSE;
+
+	gtk_tree_selection_select_path(GTK_TREE_SELECTION(
+			gtk_tree_view_get_selection(GTK_TREE_VIEW(gtkchat->list))), path);
+
+	gtk_tree_model_get_iter(GTK_TREE_MODEL(model), &iter, path);
+	gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_NAME_COLUMN, &who, -1);
+
+	if (event->button == 1 && event->type == GDK_2BUTTON_PRESS) {
+		chat_do_im(gtkconv, who);
+	} else if (event->button == 2 && event->type == GDK_BUTTON_PRESS) {
+		/* Move to user's anchor */
+		GtkTextMark *mark = get_mark_for_user(gtkconv, who);
+
+		if(mark != NULL)
+			gtk_text_view_scroll_to_mark(GTK_TEXT_VIEW(gtkconv->imhtml), mark, 0.1, FALSE, 0, 0);
+	} else if (event->button == 3 && event->type == GDK_BUTTON_PRESS) {
+		GtkWidget *menu = create_chat_menu (conv, who, gc);
+		gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL,
+					   event->button, event->time);
+	}
+
+	g_free(who);
+	gtk_tree_path_free(path);
+
+	return TRUE;
+}
+
+static void
+move_to_next_unread_tab(GaimGtkConversation *gtkconv, gboolean forward)
+{
+	GaimGtkConversation *next_gtkconv = NULL;
+	GaimGtkWindow *win;
+	int initial, i, total, diff;
+
+	win   = gtkconv->win;
+	initial = gtk_notebook_page_num(GTK_NOTEBOOK(win->notebook),
+	                                gtkconv->tab_cont);
+	total = gaim_gtk_conv_window_get_gtkconv_count(win);
+	/* By adding total here, the moduli calculated later will always have two
+	 * positive arguments. x % y where x < 0 is not guaranteed to return a
+	 * positive number.
+	 */
+	diff = (forward ? 1 : -1) + total;
+
+	for (i = (initial + diff) % total; i != initial; i = (i + diff) % total) {
+		next_gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win, i);
+		if (next_gtkconv->unseen_state > 0)
+			break;
+	}
+
+	if (i == initial) { /* no new messages */
+		i = (i + diff) % total;
+		next_gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win, i);
+	}
+
+	if (next_gtkconv != NULL && next_gtkconv != gtkconv)
+		gaim_gtk_conv_window_switch_gtkconv(win, next_gtkconv);
+}
+
+static gboolean
+entry_key_press_cb(GtkWidget *entry, GdkEventKey *event, gpointer data)
+{
+	GaimGtkWindow *win;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	int curconv;
+
+	gtkconv  = (GaimGtkConversation *)data;
+	conv     = gtkconv->active_conv;
+	win      = gtkconv->win;
+	curconv = gtk_notebook_get_current_page(GTK_NOTEBOOK(win->notebook));
+
+	/* If CTRL was held down... */
+	if (event->state & GDK_CONTROL_MASK) {
+		switch (event->keyval) {
+			case GDK_Up:
+				if (!gtkconv->send_history)
+					break;
+
+				if (!gtkconv->send_history->prev) {
+					GtkTextIter start, end;
+
+					g_free(gtkconv->send_history->data);
+
+					gtk_text_buffer_get_start_iter(gtkconv->entry_buffer,
+												   &start);
+					gtk_text_buffer_get_end_iter(gtkconv->entry_buffer, &end);
+
+					gtkconv->send_history->data =
+						gtk_imhtml_get_markup(GTK_IMHTML(gtkconv->entry));
+				}
+
+				if (gtkconv->send_history->next && gtkconv->send_history->next->data) {
+					GObject *object;
+					GtkTextIter iter;
+					GtkTextBuffer *buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->entry));
+
+					gtkconv->send_history = gtkconv->send_history->next;
+
+					/* Block the signal to prevent application of default formatting. */
+					object = g_object_ref(G_OBJECT(gtkconv->entry));
+					g_signal_handlers_block_matched(object, G_SIGNAL_MATCH_DATA, 0, 0, NULL,
+													NULL, gtkconv);
+					/* Clear the formatting. */
+					gtk_imhtml_clear_formatting(GTK_IMHTML(gtkconv->entry));
+					/* Unblock the signal. */
+					g_signal_handlers_unblock_matched(object, G_SIGNAL_MATCH_DATA, 0, 0, NULL,
+													  NULL, gtkconv);
+					g_object_unref(object);
+
+					gtk_imhtml_clear(GTK_IMHTML(gtkconv->entry));
+					gtk_imhtml_append_text_with_images(
+						GTK_IMHTML(gtkconv->entry), gtkconv->send_history->data,
+						0, NULL);
+					/* this is mainly just a hack so the formatting at the
+					 * cursor gets picked up. */
+					gtk_text_buffer_get_end_iter(buffer, &iter);
+					gtk_text_buffer_move_mark_by_name(buffer, "insert", &iter);
+				}
+
+				return TRUE;
+				break;
+
+			case GDK_Down:
+				if (!gtkconv->send_history)
+					break;
+
+				if (gtkconv->send_history->prev && gtkconv->send_history->prev->data) {
+					GObject *object;
+					GtkTextIter iter;
+					GtkTextBuffer *buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->entry));
+
+					gtkconv->send_history = gtkconv->send_history->prev;
+
+					/* Block the signal to prevent application of default formatting. */
+					object = g_object_ref(G_OBJECT(gtkconv->entry));
+					g_signal_handlers_block_matched(object, G_SIGNAL_MATCH_DATA, 0, 0, NULL,
+													NULL, gtkconv);
+					/* Clear the formatting. */
+					gtk_imhtml_clear_formatting(GTK_IMHTML(gtkconv->entry));
+					/* Unblock the signal. */
+					g_signal_handlers_unblock_matched(object, G_SIGNAL_MATCH_DATA, 0, 0, NULL,
+													  NULL, gtkconv);
+					g_object_unref(object);
+
+					gtk_imhtml_clear(GTK_IMHTML(gtkconv->entry));
+					gtk_imhtml_append_text_with_images(
+						GTK_IMHTML(gtkconv->entry), gtkconv->send_history->data,
+						0, NULL);
+					/* this is mainly just a hack so the formatting at the
+					 * cursor gets picked up. */
+					if (*(char *)gtkconv->send_history->data) {
+						gtk_text_buffer_get_end_iter(buffer, &iter);
+						gtk_text_buffer_move_mark_by_name(buffer, "insert", &iter);
+					} else {
+						/* Restore the default formatting */
+						default_formatize(gtkconv);
+					}
+				}
+
+				return TRUE;
+				break;
+
+			case GDK_Page_Down:
+			case ']':
+				if (!gaim_gtk_conv_window_get_gtkconv_at_index(win, curconv + 1))
+					gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), 0);
+				else
+					gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), curconv + 1);
+				return TRUE;
+				break;
+
+			case GDK_Page_Up:
+			case '[':
+				if (!gaim_gtk_conv_window_get_gtkconv_at_index(win, curconv - 1))
+					gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), -1);
+				else
+					gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), curconv - 1);
+				return TRUE;
+				break;
+
+			case GDK_Tab:
+			case GDK_ISO_Left_Tab:
+				if (event->state & GDK_SHIFT_MASK) {
+					move_to_next_unread_tab(gtkconv, FALSE);
+				} else {
+					move_to_next_unread_tab(gtkconv, TRUE);
+				}
+
+				return TRUE;
+				break;
+
+			case GDK_comma:
+				gtk_notebook_reorder_child(GTK_NOTEBOOK(win->notebook),
+						gtk_notebook_get_nth_page(GTK_NOTEBOOK(win->notebook), curconv),
+						curconv - 1);
+				break;
+
+			case GDK_period:
+				gtk_notebook_reorder_child(GTK_NOTEBOOK(win->notebook),
+						gtk_notebook_get_nth_page(GTK_NOTEBOOK(win->notebook), curconv),
+#if GTK_CHECK_VERSION(2,2,0)
+						(curconv + 1) % gtk_notebook_get_n_pages(GTK_NOTEBOOK(win->notebook)));
+#else
+						(curconv + 1) % g_list_length(GTK_NOTEBOOK(win->notebook)->children));
+#endif
+				break;
+
+		} /* End of switch */
+	}
+
+	/* If ALT (or whatever) was held down... */
+	else if (event->state & GDK_MOD1_MASK)
+	{
+		if (event->keyval > '0' && event->keyval <= '9')
+		{
+			guint switchto = event->keyval - '1';
+			if (switchto < gaim_gtk_conv_window_get_gtkconv_count(win))
+				gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), switchto);
+
+			return TRUE;
+		}
+	}
+
+	/* If neither CTRL nor ALT were held down... */
+	else
+	{
+		switch (event->keyval)
+		{
+			case GDK_Tab:
+				return tab_complete(conv);
+				break;
+
+			case GDK_Page_Up:
+				gtk_imhtml_page_up(GTK_IMHTML(gtkconv->imhtml));
+				return TRUE;
+				break;
+
+			case GDK_Page_Down:
+				gtk_imhtml_page_down(GTK_IMHTML(gtkconv->imhtml));
+				return TRUE;
+				break;
+
+		}
+	}
+	return FALSE;
+}
+
+/*
+ * NOTE:
+ *   This guy just kills a single right click from being propagated any
+ *   further.  I  have no idea *why* we need this, but we do ...  It
+ *   prevents right clicks on the GtkTextView in a convo dialog from
+ *   going all the way down to the notebook.  I suspect a bug in
+ *   GtkTextView, but I'm not ready to point any fingers yet.
+ */
+static gboolean
+entry_stop_rclick_cb(GtkWidget *widget, GdkEventButton *event, gpointer data)
+{
+	if (event->button == 3 && event->type == GDK_BUTTON_PRESS) {
+		/* Right single click */
+		g_signal_stop_emission_by_name(G_OBJECT(widget), "button_press_event");
+
+		return TRUE;
+	}
+
+	return FALSE;
+}
+
+/*
+ * If someone tries to type into the conversation backlog of a
+ * conversation window then we yank focus from the conversation backlog
+ * and give it to the text entry box so that people can type
+ * all the live long day and it will get entered into the entry box.
+ */
+static gboolean
+refocus_entry_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
+{
+	GaimGtkConversation *gtkconv = data;
+
+	/* If we have a valid key for the conversation display, then exit */
+	if ((event->state & GDK_CONTROL_MASK) ||
+		(event->keyval == GDK_F10) ||
+		(event->keyval == GDK_Shift_L) ||
+		(event->keyval == GDK_Shift_R) ||
+		(event->keyval == GDK_Escape) ||
+		(event->keyval == GDK_Up) ||
+		(event->keyval == GDK_Down) ||
+		(event->keyval == GDK_Left) ||
+		(event->keyval == GDK_Right) ||
+		(event->keyval == GDK_Home) ||
+		(event->keyval == GDK_End) ||
+		(event->keyval == GDK_Tab) ||
+		(event->keyval == GDK_ISO_Left_Tab))
+			return FALSE;
+
+	if (event->type == GDK_KEY_RELEASE)
+		gtk_widget_grab_focus(gtkconv->entry);
+
+	gtk_widget_event(gtkconv->entry, (GdkEvent *)event);
+
+	return TRUE;
+}
+
+void
+gaim_gtkconv_switch_active_conversation(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv;
+	GaimConversation *old_conv;
+	GtkIMHtml *entry;
+	const char *protocol_name;
+
+	g_return_if_fail(conv != NULL);
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	old_conv = gtkconv->active_conv;
+
+	if (old_conv == conv)
+		return;
+
+	gaim_conversation_close_logs(old_conv);
+	gtkconv->active_conv = conv;
+
+	gaim_conversation_set_logging(conv,
+		gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(gtkconv->win->menu.logging)));
+
+	entry = GTK_IMHTML(gtkconv->entry);
+	protocol_name = gaim_account_get_protocol_name(conv->account);
+	gtk_imhtml_set_protocol_name(entry, protocol_name);
+	gtk_imhtml_set_protocol_name(GTK_IMHTML(gtkconv->imhtml), protocol_name);
+
+	if (!(conv->features & GAIM_CONNECTION_HTML))
+		gtk_imhtml_clear_formatting(GTK_IMHTML(gtkconv->entry));
+	else if (conv->features & GAIM_CONNECTION_FORMATTING_WBFO &&
+	         !(old_conv->features & GAIM_CONNECTION_FORMATTING_WBFO))
+	{
+		/* The old conversation allowed formatting on parts of the
+		 * buffer, but the new one only allows it on the whole
+		 * buffer.  This code saves the formatting from the current
+		 * position of the cursor, clears the formatting, then
+		 * applies the saved formatting to the entire buffer. */
+
+		gboolean bold;
+		gboolean italic;
+		gboolean underline;
+		char *fontface   = gtk_imhtml_get_current_fontface(entry);
+		char *forecolor  = gtk_imhtml_get_current_forecolor(entry);
+		char *backcolor  = gtk_imhtml_get_current_backcolor(entry);
+		char *background = gtk_imhtml_get_current_background(entry);
+		gint fontsize    = gtk_imhtml_get_current_fontsize(entry);
+		gboolean bold2;
+		gboolean italic2;
+		gboolean underline2;
+
+		gtk_imhtml_get_current_format(entry, &bold, &italic, &underline);
+
+		/* Clear existing formatting */
+		gtk_imhtml_clear_formatting(entry);
+
+		/* Apply saved formatting to the whole buffer. */
+
+		gtk_imhtml_get_current_format(entry, &bold2, &italic2, &underline2);
+
+		if (bold != bold2)
+			gtk_imhtml_toggle_bold(entry);
+
+		if (italic != italic2)
+			gtk_imhtml_toggle_italic(entry);
+
+		if (underline != underline2)
+			gtk_imhtml_toggle_underline(entry);
+
+		gtk_imhtml_toggle_fontface(entry, fontface);
+
+		if (!(conv->features & GAIM_CONNECTION_NO_FONTSIZE))
+			gtk_imhtml_font_set_size(entry, fontsize);
+
+		gtk_imhtml_toggle_forecolor(entry, forecolor);
+
+		if (!(conv->features & GAIM_CONNECTION_NO_BGCOLOR))
+		{
+			gtk_imhtml_toggle_backcolor(entry, backcolor);
+			gtk_imhtml_toggle_background(entry, background);
+		}
+
+		g_free(fontface);
+		g_free(forecolor);
+		g_free(backcolor);
+		g_free(background);
+	}
+	else
+	{
+		/* This is done in default_formatize, which is called from clear_formatting_cb,
+		 * which is (obviously) a clear_formatting signal handler.  However, if we're
+		 * here, we didn't call gtk_imhtml_clear_formatting() (because we want to
+		 * preserve the formatting exactly as it is), so we have to do this now. */
+		gtk_imhtml_set_whole_buffer_formatting_only(entry,
+			(conv->features & GAIM_CONNECTION_FORMATTING_WBFO));
+	}
+
+	gaim_signal_emit(gaim_gtk_conversations_get_handle(), "conversation-switched", conv);
+
+	gray_stuff_out(gtkconv);
+	update_typing_icon(gtkconv);
+
+	gtk_window_set_title(GTK_WINDOW(gtkconv->win->window),
+	                     gtk_label_get_text(GTK_LABEL(gtkconv->tab_label)));
+}
+
+static void
+menu_conv_sel_send_cb(GObject *m, gpointer data)
+{
+	GaimAccount *account = g_object_get_data(m, "gaim_account");
+	gchar *name = g_object_get_data(m, "gaim_buddy_name");
+	GaimConversation *conv;
+
+	if (gtk_check_menu_item_get_active((GtkCheckMenuItem*) m) == FALSE)
+		return;
+
+	conv = gaim_conversation_new(GAIM_CONV_TYPE_IM, account, name);
+	gaim_gtkconv_switch_active_conversation(conv);
+}
+
+static void
+insert_text_cb(GtkTextBuffer *textbuffer, GtkTextIter *position,
+			   gchar *new_text, gint new_text_length, gpointer user_data)
+{
+	GaimGtkConversation *gtkconv = (GaimGtkConversation *)user_data;
+	GaimConversation *conv;
+
+	g_return_if_fail(gtkconv != NULL);
+
+	conv = gtkconv->active_conv;
+
+	if (!gaim_prefs_get_bool("/core/conversations/im/send_typing"))
+		return;
+
+	got_typing_keypress(gtkconv, (gtk_text_iter_is_start(position) &&
+	                    gtk_text_iter_is_end(position)));
+}
+
+static void
+delete_text_cb(GtkTextBuffer *textbuffer, GtkTextIter *start_pos,
+			   GtkTextIter *end_pos, gpointer user_data)
+{
+	GaimGtkConversation *gtkconv = (GaimGtkConversation *)user_data;
+	GaimConversation *conv;
+	GaimConvIm *im;
+
+	g_return_if_fail(gtkconv != NULL);
+
+	conv = gtkconv->active_conv;
+
+	if (!gaim_prefs_get_bool("/core/conversations/im/send_typing"))
+		return;
+
+	im = GAIM_CONV_IM(conv);
+
+	if (gtk_text_iter_is_start(start_pos) && gtk_text_iter_is_end(end_pos)) {
+
+		/* We deleted all the text, so turn off typing. */
+		gaim_conv_im_stop_send_typed_timeout(im);
+
+		serv_send_typing(gaim_conversation_get_gc(conv),
+						 gaim_conversation_get_name(conv),
+						 GAIM_NOT_TYPING);
+	}
+	else {
+		/* We're deleting, but not all of it, so it counts as typing. */
+		got_typing_keypress(gtkconv, FALSE);
+	}
+}
+
+/**************************************************************************
+ * A bunch of buddy icon functions
+ **************************************************************************/
+GdkPixbuf *
+gaim_gtkconv_get_tab_icon(GaimConversation *conv, gboolean small_icon)
+{
+	GaimAccount *account = NULL;
+	const char *name = NULL;
+	GdkPixbuf *status = NULL;
+	GaimBlistUiOps *ops = gaim_blist_get_ui_ops();
+
+	g_return_val_if_fail(conv != NULL, NULL);
+
+	account = gaim_conversation_get_account(conv);
+	name = gaim_conversation_get_name(conv);
+
+	g_return_val_if_fail(account != NULL, NULL);
+	g_return_val_if_fail(name != NULL, NULL);
+
+	/* Use the buddy icon, if possible */
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		GaimBuddy *b = gaim_find_buddy(account, name);
+		if (b != NULL) {
+			/* I hate this hack.  It fixes a bug where the pending message icon
+			 * displays in the conv tab even though it shouldn't.
+			 * A better solution would be great. */
+			if (ops && ops->update)
+				ops->update(NULL, (GaimBlistNode*)b);
+
+			status = gaim_gtk_blist_get_status_icon((GaimBlistNode*)b,
+				(small_icon ? GAIM_STATUS_ICON_SMALL : GAIM_STATUS_ICON_LARGE));
+		}
+	}
+
+	/* If they don't have a buddy icon, then use the PRPL icon */
+	if (status == NULL)
+		status = gaim_gtk_create_prpl_icon(account, small_icon ? 0.5 : 1.0);
+
+	return status;
+}
+
+static void
+update_tab_icon(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+	GdkPixbuf *status = NULL;
+
+	g_return_if_fail(conv != NULL);
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	win = gtkconv->win;
+	if (conv != gtkconv->active_conv)
+		return;
+
+	status = gaim_gtkconv_get_tab_icon(conv, TRUE);
+
+	g_return_if_fail(status != NULL);
+
+	gtk_image_set_from_pixbuf(GTK_IMAGE(gtkconv->icon), status);
+	gtk_image_set_from_pixbuf(GTK_IMAGE(gtkconv->menu_icon), status);
+
+	if (status != NULL)
+		g_object_unref(status);
+
+	if (gaim_gtk_conv_window_is_active_conversation(conv) &&
+		(gaim_conversation_get_type(conv) != GAIM_CONV_TYPE_IM ||
+		 gtkconv->u.im->anim == NULL))
+	{
+		status = gaim_gtkconv_get_tab_icon(conv, FALSE);
+
+		gtk_window_set_icon(GTK_WINDOW(win->window), status);
+
+		if (status != NULL)
+			g_object_unref(status);
+	}
+}
+
+/* This gets added as an idle handler when doing something that
+ * redraws the icon. It sets the auto_resize gboolean to TRUE.
+ * This way, when the size_allocate callback gets triggered, it notices
+ * that this is an autoresize, and after the main loop iterates, it
+ * gets set back to FALSE
+ */
+static gboolean reset_auto_resize_cb(gpointer data)
+{
+	GaimGtkConversation *gtkconv = (GaimGtkConversation *)data;
+	gtkconv->auto_resize = FALSE;
+	return FALSE;
+}
+
+static gboolean
+redraw_icon(gpointer data)
+{
+	GaimGtkConversation *gtkconv = (GaimGtkConversation *)data;
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimAccount *account;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+
+	GdkPixbuf *buf;
+	GdkPixbuf *scale;
+	gint delay;
+	int scale_width, scale_height;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	account = gaim_conversation_get_account(conv);
+	if(account && account->gc)
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(account->gc->prpl);
+
+	gtkconv->auto_resize = TRUE;
+	g_idle_add(reset_auto_resize_cb, gtkconv);
+
+	gdk_pixbuf_animation_iter_advance(gtkconv->u.im->iter, NULL);
+	buf = gdk_pixbuf_animation_iter_get_pixbuf(gtkconv->u.im->iter);
+
+	gaim_gtk_buddy_icon_get_scale_size(buf, &prpl_info->icon_spec,
+			GAIM_ICON_SCALE_DISPLAY, &scale_width, &scale_height);
+
+	/* this code is ugly, and scares me */
+	scale = gdk_pixbuf_scale_simple(buf,
+		MAX(gdk_pixbuf_get_width(buf) * scale_width /
+		    gdk_pixbuf_animation_get_width(gtkconv->u.im->anim), 1),
+		MAX(gdk_pixbuf_get_height(buf) * scale_height /
+		    gdk_pixbuf_animation_get_height(gtkconv->u.im->anim), 1),
+		GDK_INTERP_BILINEAR);
+
+	gtk_image_set_from_pixbuf(GTK_IMAGE(gtkconv->u.im->icon), scale);
+	g_object_unref(G_OBJECT(scale));
+	gtk_widget_queue_draw(gtkconv->u.im->icon);
+
+	delay = gdk_pixbuf_animation_iter_get_delay_time(gtkconv->u.im->iter);
+
+	if (delay < 100)
+		delay = 100;
+
+	gtkconv->u.im->icon_timer = g_timeout_add(delay, redraw_icon, gtkconv);
+
+	return FALSE;
+}
+
+static void
+start_anim(GtkObject *obj, GaimGtkConversation *gtkconv)
+{
+	int delay;
+
+	if (gtkconv->u.im->anim == NULL)
+		return;
+
+	if (gtkconv->u.im->icon_timer != 0)
+		return;
+
+	if (gdk_pixbuf_animation_is_static_image(gtkconv->u.im->anim))
+		return;
+
+	delay = gdk_pixbuf_animation_iter_get_delay_time(gtkconv->u.im->iter);
+
+	if (delay < 100)
+		delay = 100;
+
+	gtkconv->u.im->icon_timer = g_timeout_add(delay, redraw_icon, gtkconv);
+}
+
+static void
+remove_icon(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkWindow *gtkwin;
+
+	g_return_if_fail(conv != NULL);
+
+	if (gtkconv->u.im->icon_container != NULL)
+		gtk_widget_destroy(gtkconv->u.im->icon_container);
+
+	if (gtkconv->u.im->anim != NULL)
+		g_object_unref(G_OBJECT(gtkconv->u.im->anim));
+
+	if (gtkconv->u.im->icon_timer != 0)
+		g_source_remove(gtkconv->u.im->icon_timer);
+
+	if (gtkconv->u.im->iter != NULL)
+		g_object_unref(G_OBJECT(gtkconv->u.im->iter));
+
+	gtkconv->u.im->icon_timer = 0;
+	gtkconv->u.im->icon = NULL;
+	gtkconv->u.im->anim = NULL;
+	gtkconv->u.im->iter = NULL;
+	gtkconv->u.im->icon_container = NULL;
+	gtkconv->u.im->show_icon = FALSE;
+
+	gtkwin = gtkconv->win;
+	gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtkwin->menu.show_icon), FALSE);
+}
+
+static void
+saveicon_writefile_cb(void *user_data, const char *filename)
+{
+	GaimGtkConversation *gtkconv = (GaimGtkConversation *)user_data;
+	GaimConversation *conv = gtkconv->active_conv;
+	FILE *fp;
+	GaimBuddyIcon *icon;
+	const void *data;
+	size_t len;
+
+	if ((fp = g_fopen(filename, "wb")) == NULL) {
+		gaim_notify_error(gtkconv, NULL, _("Unable to open file."), NULL);
+		return;
+	}
+
+	icon = gaim_conv_im_get_icon(GAIM_CONV_IM(conv));
+	data = gaim_buddy_icon_get_data(icon, &len);
+
+	if ((len <= 0) || (data == NULL) || (fwrite(data, 1, len, fp) != len)) {
+		gaim_notify_error(gtkconv, NULL, _("Unable to save icon file to disk."), NULL);
+		fclose(fp);
+		g_unlink(filename);
+		return;
+	}
+	fclose(fp);
+}
+
+static const char *
+custom_icon_pref_name(GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv;
+	GaimAccount *account;
+	GaimBuddy *buddy;
+
+	conv = gtkconv->active_conv;
+	account = gaim_conversation_get_account(conv);
+	buddy = gaim_find_buddy(account, gaim_conversation_get_name(conv));
+	if (buddy) {
+		GaimContact *contact = gaim_buddy_get_contact(buddy);
+		return gaim_blist_node_get_string((GaimBlistNode*)contact, "custom_buddy_icon");
+	}
+	return NULL;
+}
+
+static void
+custom_icon_sel_cb(const char *filename, gpointer data)
+{
+	if (filename) {
+		GaimGtkConversation *gtkconv = data;
+		GaimConversation *conv = gtkconv->active_conv;
+		GaimAccount *account = gaim_conversation_get_account(conv);
+		gaim_gtk_set_custom_buddy_icon(account, gaim_conversation_get_name(conv), filename);
+	}
+}
+
+static void
+set_custom_icon_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GtkWidget *win = gaim_gtk_buddy_icon_chooser_new(GTK_WINDOW(gtkconv->win->window),
+						custom_icon_sel_cb, gtkconv);
+	gtk_widget_show_all(win);
+}
+
+static void
+remove_custom_icon_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv;
+	GaimAccount *account;
+
+	conv = gtkconv->active_conv;
+	account = gaim_conversation_get_account(conv);
+	gaim_gtk_set_custom_buddy_icon(account, gaim_conversation_get_name(conv), NULL);
+}
+
+static void
+icon_menu_save_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	const gchar *ext;
+	gchar *buf;
+
+	g_return_if_fail(conv != NULL);
+
+	ext = gaim_buddy_icon_get_type(gaim_conv_im_get_icon(GAIM_CONV_IM(conv)));
+	if (ext == NULL)
+		ext = "icon";
+
+	buf = g_strdup_printf("%s.%s", gaim_normalize(conv->account, conv->name), ext);
+
+	gaim_request_file(gtkconv, _("Save Icon"), buf, TRUE,
+					 G_CALLBACK(saveicon_writefile_cb), NULL, gtkconv);
+
+	g_free(buf);
+}
+
+static void
+stop_anim(GtkObject *obj, GaimGtkConversation *gtkconv)
+{
+	if (gtkconv->u.im->icon_timer != 0)
+		g_source_remove(gtkconv->u.im->icon_timer);
+
+	gtkconv->u.im->icon_timer = 0;
+}
+
+
+static void
+toggle_icon_animate_cb(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	gtkconv->u.im->animate =
+		gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(w));
+
+	if (gtkconv->u.im->animate)
+		start_anim(NULL, gtkconv);
+	else
+		stop_anim(NULL, gtkconv);
+}
+
+static gboolean
+icon_menu(GtkObject *obj, GdkEventButton *e, GaimGtkConversation *gtkconv)
+{
+	static GtkWidget *menu = NULL;
+	const char *pref;
+
+	if (e->button != 3 || e->type != GDK_BUTTON_PRESS)
+		return FALSE;
+
+	/*
+	 * If a menu already exists, destroy it before creating a new one,
+	 * thus freeing-up the memory it occupied.
+	 */
+	if (menu != NULL)
+		gtk_widget_destroy(menu);
+
+	menu = gtk_menu_new();
+
+	if (gtkconv->u.im->anim &&
+		!(gdk_pixbuf_animation_is_static_image(gtkconv->u.im->anim)))
+	{
+		gaim_new_check_item(menu, _("Animate"),
+							G_CALLBACK(toggle_icon_animate_cb), gtkconv,
+							gtkconv->u.im->icon_timer);
+	}
+
+	gaim_new_item_from_stock(menu, _("Hide Icon"), NULL, G_CALLBACK(remove_icon),
+							 gtkconv, 0, 0, NULL);
+
+	gaim_new_item_from_stock(menu, _("Save Icon As..."), GTK_STOCK_SAVE_AS,
+							 G_CALLBACK(icon_menu_save_cb), gtkconv,
+							 0, 0, NULL);
+
+	gaim_new_item_from_stock(menu, _("Set Custom Icon..."), NULL,
+							 G_CALLBACK(set_custom_icon_cb), gtkconv,
+							 0, 0, NULL);
+
+	/* Is there a custom icon for this person? */
+	pref = custom_icon_pref_name(gtkconv);
+	if (pref && *pref) {
+		gaim_new_item_from_stock(menu, _("Remove Custom Icon"), NULL,
+							G_CALLBACK(remove_custom_icon_cb), gtkconv,
+							0, 0, NULL);
+	}
+
+	gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, e->button, e->time);
+
+	return TRUE;
+}
+
+static void
+menu_buddyicon_cb(gpointer data, guint action, GtkWidget *widget)
+{
+	GaimGtkWindow *win = data;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	gboolean active;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (!conv)
+		return;
+
+	g_return_if_fail(gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM);
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	active = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
+	gtkconv->u.im->show_icon = active;
+	if (active)
+		gaim_gtkconv_update_buddy_icon(conv);
+	else
+		remove_icon(NULL, gtkconv);
+}
+
+/**************************************************************************
+ * End of the bunch of buddy icon functions
+ **************************************************************************/
+void
+gaim_gtkconv_present_conversation(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	if(gtkconv->win==hidden_convwin) {
+		gaim_gtk_conv_window_remove_gtkconv(hidden_convwin, gtkconv);
+		gaim_gtkconv_placement_place(gtkconv);
+	}
+
+	gaim_gtkconv_switch_active_conversation(conv);
+	gaim_gtk_conv_window_switch_gtkconv(gtkconv->win, gtkconv);
+	gtk_window_present(GTK_WINDOW(gtkconv->win->window));
+}
+
+GList *
+gaim_gtk_conversations_find_unseen_list(GaimConversationType type,
+										GaimUnseenState min_state,
+										gboolean hidden_only,
+										guint max_count)
+{
+	GList *l;
+	GList *r = NULL;
+	guint c = 0;
+
+	if (type == GAIM_CONV_TYPE_IM) {
+		l = gaim_get_ims();
+	} else if (type == GAIM_CONV_TYPE_CHAT) {
+		l = gaim_get_chats();
+	} else {
+		l = gaim_get_conversations();
+	}
+
+	for (; l != NULL && (max_count == 0 || c < max_count); l = l->next) {
+		GaimConversation *conv = (GaimConversation*)l->data;
+		GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+		if(gtkconv->active_conv != conv)
+			continue;
+
+		if (gtkconv->unseen_state >= min_state
+			&& (!hidden_only ||
+				(hidden_only && gtkconv->win == hidden_convwin))) {
+
+			r = g_list_prepend(r, conv);
+			c++;
+		}
+	}
+
+	return r;
+}
+
+static void
+unseen_conv_menu_cb(GtkMenuItem *item, GaimConversation *conv)
+{
+	g_return_if_fail(conv != NULL);
+	gaim_gtkconv_present_conversation(conv);
+}
+
+guint
+gaim_gtk_conversations_fill_menu(GtkWidget *menu, GList *convs)
+{
+	GList *l;
+	guint ret=0;
+
+	g_return_val_if_fail(menu != NULL, 0);
+	g_return_val_if_fail(convs != NULL, 0);
+
+	for (l = convs; l != NULL ; l = l->next) {
+		GaimConversation *conv = (GaimConversation*)l->data;
+		GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+		GtkWidget *icon = gtk_image_new();
+		GdkPixbuf *pbuf = gaim_gtkconv_get_tab_icon(conv, TRUE);
+		GtkWidget *item;
+		gchar *text = g_strdup_printf("%s (%d)",
+				gtk_label_get_text(GTK_LABEL(gtkconv->tab_label)),
+				gtkconv->unseen_count);
+
+		gtk_image_set_from_pixbuf(GTK_IMAGE(icon), pbuf);
+		g_object_unref(pbuf);
+
+		item = gtk_image_menu_item_new_with_label(text);
+		gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), icon);
+		g_signal_connect(G_OBJECT(item), "activate", G_CALLBACK(unseen_conv_menu_cb), conv);
+		gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+		g_free(text);
+		ret++;
+	}
+
+	return ret;
+}
+
+GaimGtkWindow *
+gaim_gtkconv_get_window(GaimGtkConversation *gtkconv)
+{
+	g_return_val_if_fail(gtkconv != NULL, NULL);
+	return gtkconv->win;
+}
+
+static GtkItemFactoryEntry menu_items[] =
+{
+	/* Conversation menu */
+	{ N_("/_Conversation"), NULL, NULL, 0, "<Branch>", NULL },
+
+	{ N_("/Conversation/New Instant _Message..."), "<CTL>M", menu_new_conv_cb,
+			0, "<StockItem>", GAIM_STOCK_IM },
+
+	{ "/Conversation/sep0", NULL, NULL, 0, "<Separator>", NULL },
+
+	{ N_("/Conversation/_Find..."), NULL, menu_find_cb, 0,
+			"<StockItem>", GTK_STOCK_FIND },
+	{ N_("/Conversation/View _Log"), NULL, menu_view_log_cb, 0, "<StockItem>", GAIM_STOCK_LOG },
+	{ N_("/Conversation/_Save As..."), NULL, menu_save_as_cb, 0,
+			"<StockItem>", GTK_STOCK_SAVE_AS },
+	{ N_("/Conversation/Clea_r Scrollback"), "<CTL>L", menu_clear_cb, 0, "<StockItem>", GTK_STOCK_CLEAR },
+
+	{ "/Conversation/sep1", NULL, NULL, 0, "<Separator>", NULL },
+
+	{ N_("/Conversation/Se_nd File..."), NULL, menu_send_file_cb, 0, "<StockItem>", GAIM_STOCK_FILE_TRANSFER },
+	{ N_("/Conversation/Add Buddy _Pounce..."), NULL, menu_add_pounce_cb,
+			0, "<StockItem>", GAIM_STOCK_POUNCE },
+	{ N_("/Conversation/_Get Info"), "<CTL>O", menu_get_info_cb, 0,
+			"<StockItem>", GAIM_STOCK_INFO },
+	{ N_("/Conversation/In_vite..."), NULL, menu_invite_cb, 0,
+			"<StockItem>", GAIM_STOCK_INVITE },
+	{ N_("/Conversation/M_ore"), NULL, NULL, 0, "<Branch>", NULL },
+
+	{ "/Conversation/sep2", NULL, NULL, 0, "<Separator>", NULL },
+
+	{ N_("/Conversation/Al_ias..."), NULL, menu_alias_cb, 0,
+			"<StockItem>", GAIM_STOCK_EDIT },
+	{ N_("/Conversation/_Block..."), NULL, menu_block_cb, 0,
+			"<StockItem>", GAIM_STOCK_BLOCK },
+	{ N_("/Conversation/_Add..."), NULL, menu_add_remove_cb, 0,
+			"<StockItem>", GTK_STOCK_ADD },
+	{ N_("/Conversation/_Remove..."), NULL, menu_add_remove_cb, 0,
+			"<StockItem>", GTK_STOCK_REMOVE },
+
+	{ "/Conversation/sep3", NULL, NULL, 0, "<Separator>", NULL },
+
+	{ N_("/Conversation/Insert Lin_k..."), NULL, menu_insert_link_cb, 0,
+			"<StockItem>", GAIM_STOCK_LINK },
+	{ N_("/Conversation/Insert Imag_e..."), NULL, menu_insert_image_cb, 0,
+			"<StockItem>", GAIM_STOCK_IMAGE },
+
+	{ "/Conversation/sep4", NULL, NULL, 0, "<Separator>", NULL },
+
+	{ N_("/Conversation/_Close"), NULL, menu_close_conv_cb, 0,
+			"<StockItem>", GTK_STOCK_CLOSE },
+
+	/* Options */
+	{ N_("/_Options"), NULL, NULL, 0, "<Branch>", NULL },
+	{ N_("/Options/Enable _Logging"), NULL, menu_logging_cb, 0, "<CheckItem>", NULL },
+	{ N_("/Options/Enable _Sounds"), NULL, menu_sounds_cb, 0, "<CheckItem>", NULL },
+	{ N_("/Options/Show Buddy _Icon"), NULL, menu_buddyicon_cb, 0, "<CheckItem>", NULL },
+	{ "/Options/sep0", NULL, NULL, 0, "<Separator>", NULL },
+	{ N_("/Options/Show Formatting _Toolbars"), NULL, menu_toolbar_cb, 0, "<CheckItem>", NULL },
+	{ N_("/Options/Show Ti_mestamps"), "F2", menu_timestamps_cb, 0, "<CheckItem>", NULL },
+};
+
+static const int menu_item_count =
+sizeof(menu_items) / sizeof(*menu_items);
+
+static const char *
+item_factory_translate_func (const char *path, gpointer func_data)
+{
+	return _(path);
+}
+
+static void
+sound_method_pref_changed_cb(const char *name, GaimPrefType type,
+							 gconstpointer value, gpointer data)
+{
+	GaimGtkWindow *win = data;
+	const char *method = value;
+
+	if (!strcmp(method, "none"))
+	{
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.sounds),
+		                               FALSE);
+		gtk_widget_set_sensitive(win->menu.sounds, FALSE);
+	}
+	else
+	{
+		GaimGtkConversation *gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+
+		if (gtkconv != NULL)
+			gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.sounds),
+			                               TRUE);
+		gtk_widget_set_sensitive(win->menu.sounds, TRUE);
+
+	}
+}
+
+static void
+show_buddy_icons_pref_changed_cb(const char *name, GaimPrefType type,
+								 gconstpointer value, gpointer data)
+{
+	GaimGtkWindow *win = data;
+	gboolean show_icons = GPOINTER_TO_INT(value);
+
+	if (!show_icons)
+	{
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_icon),
+		                               FALSE);
+		gtk_widget_set_sensitive(win->menu.show_icon, FALSE);
+	}
+	else
+	{
+		GaimGtkConversation *gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+
+		if (gtkconv != NULL)
+			gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_icon),
+			                               TRUE);
+		gtk_widget_set_sensitive(win->menu.show_icon, TRUE);
+
+	}
+}
+
+static void
+regenerate_options_items(GaimGtkWindow *win)
+{
+	GtkWidget *menu;
+	GList *list;
+	GaimGtkConversation *gtkconv;
+	GaimConversation *conv;
+	GaimBuddy *buddy;
+
+	gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+	conv = gtkconv->active_conv;
+	buddy = gaim_find_buddy(conv->account, conv->name);
+
+	menu = gtk_item_factory_get_widget(win->menu.item_factory, N_("/Conversation/More"));
+
+	/* Remove the previous entries */
+	for (list = gtk_container_get_children(GTK_CONTAINER(menu)); list; )
+	{
+		GtkWidget *w = list->data;
+		list = list->next;
+		gtk_widget_destroy(w);
+	}
+
+	/* Now add the stuff */
+	if (buddy)
+	{
+		if (gaim_account_is_connected(conv->account))
+			gaim_gtk_append_blist_node_proto_menu(menu, conv->account->gc,
+												  (GaimBlistNode *)buddy);
+		gaim_gtk_append_blist_node_extended_menu(menu, (GaimBlistNode *)buddy);
+	}
+
+	if ((list = gtk_container_get_children(GTK_CONTAINER(menu))) == NULL)
+	{
+		GtkWidget *item = gtk_menu_item_new_with_label(_("No actions available"));
+		gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+		gtk_widget_set_sensitive(item, FALSE);
+	}
+
+	gtk_widget_show_all(menu);
+}
+
+static void menubar_activated(GtkWidget *item, gpointer data)
+{
+	GaimGtkWindow *win = data;
+	regenerate_options_items(win);
+
+	/* The following are to make sure the 'More' submenu is not regenerated every time
+	 * the focus shifts from 'Conversations' to some other menu and back. */
+	g_signal_handlers_block_by_func(G_OBJECT(item), G_CALLBACK(menubar_activated), data);
+	g_signal_connect(G_OBJECT(win->menu.menubar), "deactivate", G_CALLBACK(focus_out_from_menubar), data);
+}
+
+static void
+focus_out_from_menubar(GtkWidget *wid, GaimGtkWindow *win)
+{
+	/* The menubar has been deactivated. Make sure the 'More' submenu is regenerated next time
+	 * the 'Conversation' menu pops up. */
+	GtkWidget *menuitem = gtk_item_factory_get_item(win->menu.item_factory, N_("/Conversation"));
+	g_signal_handlers_unblock_by_func(G_OBJECT(menuitem), G_CALLBACK(menubar_activated), win);
+	g_signal_handlers_disconnect_by_func(G_OBJECT(win->menu.menubar),
+				G_CALLBACK(focus_out_from_menubar), win);
+}
+
+static GtkWidget *
+setup_menubar(GaimGtkWindow *win)
+{
+	GtkAccelGroup *accel_group;
+	const char *method;
+	GtkWidget *menuitem;
+
+	accel_group = gtk_accel_group_new ();
+	gtk_window_add_accel_group(GTK_WINDOW(win->window), accel_group);
+	g_object_unref(accel_group);
+
+	win->menu.item_factory =
+		gtk_item_factory_new(GTK_TYPE_MENU_BAR, "<main>", accel_group);
+
+	gtk_item_factory_set_translate_func(win->menu.item_factory,
+	                                    (GtkTranslateFunc)item_factory_translate_func,
+	                                    NULL, NULL);
+
+	gtk_item_factory_create_items(win->menu.item_factory, menu_item_count,
+	                              menu_items, win);
+	g_signal_connect(G_OBJECT(accel_group), "accel-changed",
+	                 G_CALLBACK(gaim_gtk_save_accels_cb), NULL);
+
+	/* Make sure the 'Conversation -> More' menuitems are regenerated whenever
+	 * the 'Conversation' menu pops up because the entries can change after the
+	 * conversation is created. */
+	menuitem = gtk_item_factory_get_item(win->menu.item_factory, N_("/Conversation"));
+	g_signal_connect(G_OBJECT(menuitem), "activate", G_CALLBACK(menubar_activated), win);
+
+	win->menu.menubar =
+		gtk_item_factory_get_widget(win->menu.item_factory, "<main>");
+
+	win->menu.view_log =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/View Log"));
+
+	/* --- */
+
+	win->menu.send_file =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Send File..."));
+
+	win->menu.add_pounce =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Add Buddy Pounce..."));
+
+	/* --- */
+
+	win->menu.get_info =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Get Info"));
+
+	win->menu.invite =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Invite..."));
+
+	/* --- */
+
+	win->menu.alias =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Alias..."));
+
+	win->menu.block =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Block..."));
+
+	win->menu.add =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Add..."));
+
+	win->menu.remove =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Remove..."));
+
+	/* --- */
+
+	win->menu.insert_link =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Insert Link..."));
+
+	win->menu.insert_image =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Conversation/Insert Image..."));
+
+	/* --- */
+
+	win->menu.logging =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Options/Enable Logging"));
+	win->menu.sounds =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Options/Enable Sounds"));
+	method = gaim_prefs_get_string("/gaim/gtk/sound/method");
+	if (!strcmp(method, "none"))
+	{
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.sounds),
+		                               FALSE);
+		gtk_widget_set_sensitive(win->menu.sounds, FALSE);
+	}
+	gaim_prefs_connect_callback(win, "/gaim/gtk/sound/method",
+				    sound_method_pref_changed_cb, win);
+
+	win->menu.show_formatting_toolbar =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Options/Show Formatting Toolbars"));
+	win->menu.show_timestamps =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Options/Show Timestamps"));
+	win->menu.show_icon =
+		gtk_item_factory_get_widget(win->menu.item_factory,
+		                            N_("/Options/Show Buddy Icon"));
+	if (!gaim_prefs_get_bool("/gaim/gtk/conversations/im/show_buddy_icons"))
+	{
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_icon),
+		                               FALSE);
+		gtk_widget_set_sensitive(win->menu.show_icon, FALSE);
+	}
+	gaim_prefs_connect_callback(win, "/gaim/gtk/conversations/im/show_buddy_icons",
+				    show_buddy_icons_pref_changed_cb, win);
+
+	win->menu.tray = gaim_gtk_menu_tray_new();
+	gtk_menu_shell_append(GTK_MENU_SHELL(win->menu.menubar),
+	                      win->menu.tray);
+	gtk_widget_show(win->menu.tray);
+
+	gtk_widget_show(win->menu.menubar);
+
+	return win->menu.menubar;
+}
+
+
+/**************************************************************************
+ * Utility functions
+ **************************************************************************/
+
+static void
+got_typing_keypress(GaimGtkConversation *gtkconv, gboolean first)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimConvIm *im;
+
+	/*
+	 * We know we got something, so we at least have to make sure we don't
+	 * send GAIM_TYPED any time soon.
+	 */
+
+	im = GAIM_CONV_IM(conv);
+
+	gaim_conv_im_stop_send_typed_timeout(im);
+	gaim_conv_im_start_send_typed_timeout(im);
+
+	/* Check if we need to send another GAIM_TYPING message */
+	if (first || (gaim_conv_im_get_type_again(im) != 0 &&
+				  time(NULL) > gaim_conv_im_get_type_again(im)))
+	{
+		unsigned int timeout;
+		timeout = serv_send_typing(gaim_conversation_get_gc(conv),
+								   gaim_conversation_get_name(conv),
+								   GAIM_TYPING);
+		gaim_conv_im_set_type_again(im, timeout);
+	}
+}
+
+static void
+update_typing_icon(GaimGtkConversation *gtkconv)
+{
+	GaimGtkWindow *gtkwin;
+	GaimConvIm *im = NULL;
+	GaimConversation *conv = gtkconv->active_conv;
+	char *stock_id;
+	const char *tooltip;
+
+	gtkwin = gtkconv->win;
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		im = GAIM_CONV_IM(conv);
+
+	if (gtkwin->menu.typing_icon) {
+		gtk_widget_hide(gtkwin->menu.typing_icon);
+	}
+
+	if (!im || (gaim_conv_im_get_typing_state(im) == GAIM_NOT_TYPING))
+		return;
+
+	if (gaim_conv_im_get_typing_state(im) == GAIM_TYPING) {
+		stock_id = GAIM_STOCK_TYPING;
+		tooltip = _("User is typing...");
+	} else {
+		stock_id = GAIM_STOCK_TYPED;
+		tooltip = _("User has typed something and stopped");
+	}
+
+	if (gtkwin->menu.typing_icon == NULL)
+	{
+		gtkwin->menu.typing_icon = gtk_image_new_from_stock(stock_id, GTK_ICON_SIZE_MENU);
+		gaim_gtk_menu_tray_append(GAIM_GTK_MENU_TRAY(gtkwin->menu.tray),
+								  gtkwin->menu.typing_icon,
+								  tooltip);
+	}
+	else
+	{
+		gtk_image_set_from_stock(GTK_IMAGE(gtkwin->menu.typing_icon), stock_id, GTK_ICON_SIZE_MENU);
+		gaim_gtk_menu_tray_set_tooltip(GAIM_GTK_MENU_TRAY(gtkwin->menu.tray),
+									   gtkwin->menu.typing_icon,
+									   tooltip);
+	}
+
+	gtk_widget_show(gtkwin->menu.typing_icon);
+}
+
+static gboolean
+update_send_to_selection(GaimGtkWindow *win)
+{
+	GaimAccount *account;
+	GaimConversation *conv;
+	GtkWidget *menu;
+	GList *child;
+	GaimBuddy *b;
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (conv == NULL)
+		return FALSE;
+
+	account = gaim_conversation_get_account(conv);
+
+	if (account == NULL)
+		return FALSE;
+
+	if (win->menu.send_to == NULL)
+		return FALSE;
+
+	if (!(b = gaim_find_buddy(account, conv->name)))
+		return FALSE;
+
+
+	gtk_widget_show(win->menu.send_to);
+
+	menu = gtk_menu_item_get_submenu(
+		GTK_MENU_ITEM(win->menu.send_to));
+
+	for (child = gtk_container_get_children(GTK_CONTAINER(menu));
+		 child != NULL;
+		 child = child->next) {
+
+		GtkWidget *item = child->data;
+		GaimBuddy *item_buddy;
+		GaimAccount *item_account = g_object_get_data(G_OBJECT(item), "gaim_account");
+		gchar *buddy_name = g_object_get_data(G_OBJECT(item),
+		                                      "gaim_buddy_name");
+		item_buddy = gaim_find_buddy(item_account, buddy_name);
+
+		if (b == item_buddy) {
+			gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(item), TRUE);
+			break;
+		}
+	}
+
+	return FALSE;
+}
+
+static gboolean
+send_to_item_enter_notify_cb(GtkWidget *menuitem, GdkEventCrossing *event, GtkWidget *label)
+{
+	gtk_widget_set_sensitive(GTK_WIDGET(label), TRUE);
+	return FALSE;
+}
+
+static gboolean
+send_to_item_leave_notify_cb(GtkWidget *menuitem, GdkEventCrossing *event, GtkWidget *label)
+{
+	gtk_widget_set_sensitive(GTK_WIDGET(label), FALSE);
+	return FALSE;
+}
+
+static void
+create_sendto_item(GtkWidget *menu, GtkSizeGroup *sg, GSList **group, GaimBuddy *buddy, GaimAccount *account, const char *name)
+{
+	GtkWidget *box;
+	GtkWidget *label;
+	GtkWidget *image;
+	GtkWidget *menuitem;
+	GdkPixbuf *pixbuf;
+	gchar *text;
+
+	/* Create a pixmap for the protocol icon. */
+	if (buddy != NULL)
+		pixbuf = gaim_gtk_blist_get_status_icon((GaimBlistNode*)buddy, GAIM_STATUS_ICON_SMALL);
+	else
+		pixbuf = gaim_gtk_create_prpl_icon(account, 0.5);
+
+	/* Now convert it to GtkImage */
+	if (pixbuf == NULL)
+		image = gtk_image_new();
+	else
+	{
+		image = gtk_image_new_from_pixbuf(pixbuf);
+		g_object_unref(G_OBJECT(pixbuf));
+	}
+
+	gtk_size_group_add_widget(sg, image);
+
+	/* Make our menu item */
+	text = g_strdup_printf("%s (%s)", name, gaim_account_get_username(account));
+	menuitem = gtk_radio_menu_item_new_with_label(*group, text);
+	g_free(text);
+	*group = gtk_radio_menu_item_get_group(GTK_RADIO_MENU_ITEM(menuitem));
+
+	/* Do some evil, see some evil, speak some evil. */
+	box = gtk_hbox_new(FALSE, 0);
+
+	label = gtk_bin_get_child(GTK_BIN(menuitem));
+	g_object_ref(label);
+	gtk_container_remove(GTK_CONTAINER(menuitem), label);
+
+	gtk_box_pack_start(GTK_BOX(box), image, FALSE, FALSE, 0);
+	gtk_box_pack_start(GTK_BOX(box), label, TRUE, TRUE, 4);
+
+	if (buddy != NULL &&
+	    !gaim_presence_is_online(gaim_buddy_get_presence(buddy)) &&
+	    !gaim_account_supports_offline_message(account, buddy))
+	{
+		gtk_widget_set_sensitive(label, FALSE);
+
+		/* Set the label sensitive when the menuitem is highlighted and
+		 * insensitive again when the mouse leaves it. This way, it
+		 * doesn't appear weird from the highlighting of the embossed
+		 * (insensitive style) text.*/
+		g_signal_connect(menuitem, "enter-notify-event",
+				 G_CALLBACK(send_to_item_enter_notify_cb), label);
+		g_signal_connect(menuitem, "leave-notify-event",
+				 G_CALLBACK(send_to_item_leave_notify_cb), label);
+	}
+
+	g_object_unref(label);
+
+	gtk_container_add(GTK_CONTAINER(menuitem), box);
+
+	gtk_widget_show(label);
+	gtk_widget_show(image);
+	gtk_widget_show(box);
+
+	/* Set our data and callbacks. */
+	g_object_set_data(G_OBJECT(menuitem), "gaim_account", account);
+	g_object_set_data_full(G_OBJECT(menuitem), "gaim_buddy_name", g_strdup(name), g_free);
+
+	g_signal_connect(G_OBJECT(menuitem), "activate",
+	                 G_CALLBACK(menu_conv_sel_send_cb), NULL);
+
+	gtk_widget_show(menuitem);
+	gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
+}
+
+static void
+generate_send_to_items(GaimGtkWindow *win)
+{
+	GtkWidget *menu;
+	GSList *group = NULL;
+	GtkSizeGroup *sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
+	GaimGtkConversation *gtkconv;
+	GSList *l, *buds;
+
+	g_return_if_fail(win != NULL);
+
+	gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+
+	g_return_if_fail(gtkconv != NULL);
+
+	if (win->menu.send_to != NULL)
+		gtk_widget_destroy(win->menu.send_to);
+
+	/* Build the Send To menu */
+	win->menu.send_to = gtk_menu_item_new_with_mnemonic(_("_Send To"));
+	gtk_widget_show(win->menu.send_to);
+
+	menu = gtk_menu_new();
+	gtk_menu_shell_insert(GTK_MENU_SHELL(win->menu.menubar),
+	                      win->menu.send_to, 2);
+	gtk_menu_item_set_submenu(GTK_MENU_ITEM(win->menu.send_to), menu);
+
+	gtk_widget_show(menu);
+
+	if (gtkconv->active_conv->type == GAIM_CONV_TYPE_IM) {
+		buds = gaim_find_buddies(gtkconv->active_conv->account, gtkconv->active_conv->name);
+
+		if (buds == NULL)
+		{
+			/* The user isn't on the buddy list. */
+			create_sendto_item(menu, sg, &group, NULL, gtkconv->active_conv->account, gtkconv->active_conv->name);
+		}
+		else
+		{
+			GList *list = NULL, *iter;
+			for (l = buds; l != NULL; l = l->next)
+			{
+				GaimBlistNode *node;
+
+				node = (GaimBlistNode *) gaim_buddy_get_contact((GaimBuddy *)l->data);
+
+				for (node = node->child; node != NULL; node = node->next)
+				{
+					GaimBuddy *buddy = (GaimBuddy *)node;
+					GaimAccount *account;
+
+					if (!GAIM_BLIST_NODE_IS_BUDDY(node))
+						continue;
+
+					account = gaim_buddy_get_account(buddy);
+					if (gaim_account_is_connected(account))
+					{
+						/* Use the GaimPresence to get unique buddies. */
+						GaimPresence *presence = gaim_buddy_get_presence(buddy);
+						if (!g_list_find(list, presence))
+							list = g_list_prepend(list, presence);
+					}
+				}
+			}
+
+			/* Loop over the list backwards so we get the items in the right order,
+			 * since we did a g_list_prepend() earlier. */
+			for (iter = g_list_last(list); iter != NULL; iter = iter->prev)
+			{
+				GaimPresence *pre = iter->data;
+				GaimBuddy *buddy = gaim_presence_get_buddies(pre)->data;
+				create_sendto_item(menu, sg, &group, buddy,
+							gaim_buddy_get_account(buddy), gaim_buddy_get_name(buddy));
+			}
+			g_list_free(list);
+			g_slist_free(buds);
+		}
+	}
+
+	g_object_unref(sg);
+
+	gtk_widget_show(win->menu.send_to);
+	/* TODO: This should never be insensitive.  Possibly hidden or not. */
+	if (!group)
+		gtk_widget_set_sensitive(win->menu.send_to, FALSE);
+	update_send_to_selection(win);
+}
+
+static GList *
+generate_invite_user_names(GaimConnection *gc)
+{
+	GaimBlistNode *gnode,*cnode,*bnode;
+	static GList *tmp = NULL;
+
+	g_list_free(tmp);
+
+	tmp = g_list_append(NULL, "");
+
+	if (gc != NULL) {
+		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) {
+				if(!GAIM_BLIST_NODE_IS_CONTACT(cnode))
+					continue;
+				for(bnode = cnode->child; bnode; bnode = bnode->next) {
+					GaimBuddy *buddy;
+
+					if(!GAIM_BLIST_NODE_IS_BUDDY(bnode))
+						continue;
+
+					buddy = (GaimBuddy *)bnode;
+
+					if (buddy->account == gc->account &&
+							GAIM_BUDDY_IS_ONLINE(buddy))
+						tmp = g_list_insert_sorted(tmp, buddy->name,
+												   (GCompareFunc)g_utf8_collate);
+				}
+			}
+		}
+	}
+
+	return tmp;
+}
+
+static GdkPixbuf *
+get_chat_buddy_status_icon(GaimConvChat *chat, const char *name, GaimConvChatBuddyFlags flags)
+{
+	GdkPixbuf *pixbuf, *scale, *scale2;
+	char *filename;
+	const char *image = NULL;
+
+	if (flags & GAIM_CBFLAGS_FOUNDER) {
+		image = "founder.png";
+	} else if (flags & GAIM_CBFLAGS_OP) {
+		image = "op.png";
+	} else if (flags & GAIM_CBFLAGS_HALFOP) {
+		image = "halfop.png";
+	} else if (flags & GAIM_CBFLAGS_VOICE) {
+		image = "voice.png";
+	} else if ((!flags) && gaim_conv_chat_is_user_ignored(chat, name)) {
+		image = "ignored.png";
+	} else {
+		return NULL;
+	}
+
+	filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", image, NULL);
+	pixbuf = gdk_pixbuf_new_from_file(filename, NULL);
+	g_free(filename);
+
+	if (!pixbuf)
+		return NULL;
+
+	scale = gdk_pixbuf_scale_simple(pixbuf, 15, 15, GDK_INTERP_BILINEAR);
+	g_object_unref(pixbuf);
+
+	if (flags && gaim_conv_chat_is_user_ignored(chat, name)) {
+		filename = g_build_filename(DATADIR, "pixmaps", "gaim", "status", "default", "ignored.png", NULL);
+		pixbuf = gdk_pixbuf_new_from_file(filename, NULL);
+		g_free(filename);
+		scale2 = gdk_pixbuf_scale_simple(pixbuf, 15, 15, GDK_INTERP_BILINEAR);
+		g_object_unref(pixbuf);
+		gdk_pixbuf_composite(scale2, scale, 0, 0, 15, 15, 0, 0, 1, 1, GDK_INTERP_BILINEAR, 192);
+		g_object_unref(scale2);
+	}
+
+	return scale;
+}
+
+static void
+add_chat_buddy_common(GaimConversation *conv, GaimConvChatBuddy *cb, const char *old_name)
+{
+	GaimGtkConversation *gtkconv;
+	GaimGtkChatPane *gtkchat;
+	GaimConvChat *chat;
+	GaimConnection *gc;
+	GaimPluginProtocolInfo *prpl_info;
+	GtkListStore *ls;
+	GdkPixbuf *pixbuf;
+	GtkTreeIter iter;
+	gboolean is_me = FALSE;
+	gboolean is_buddy;
+	gchar *tmp, *alias_key, *name, *alias;
+	int flags;
+
+	alias = cb->alias;
+	name  = cb->name;
+	flags = GPOINTER_TO_INT(cb->flags);
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+	gc      = gaim_conversation_get_gc(conv);
+
+	if (!gc || !(prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)))
+		return;
+
+	ls = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list)));
+
+	pixbuf = get_chat_buddy_status_icon(chat, name, flags);
+
+	if (!strcmp(chat->nick, gaim_normalize(conv->account, old_name != NULL ? old_name : name)))
+		is_me = TRUE;
+
+	is_buddy = (gaim_find_buddy(conv->account, name) != NULL);
+
+	tmp = g_utf8_casefold(alias, -1);
+	alias_key = g_utf8_collate_key(tmp, -1);
+	g_free(tmp);
+
+	if (is_me)
+	{
+		GdkColor send_color;
+		gdk_color_parse(SEND_COLOR, &send_color);
+
+#if GTK_CHECK_VERSION(2,6,0)
+		gtk_list_store_insert_with_values(ls, &iter,
+/*
+ * The GTK docs are mute about the effects of the "row" value for performance.
+ * X-Chat hardcodes their value to 0 (prepend) and -1 (append), so we will too.
+ * It *might* be faster to search the gtk_list_store and set row accurately,
+ * but no one in #gtk+ seems to know anything about it either.
+ * Inserting in the "wrong" location has no visible ill effects. - F.P.
+ */
+		                                  -1, /* "row" */
+		                                  CHAT_USERS_ICON_COLUMN,  pixbuf,
+		                                  CHAT_USERS_ALIAS_COLUMN, alias,
+		                                  CHAT_USERS_ALIAS_KEY_COLUMN, alias_key,
+		                                  CHAT_USERS_NAME_COLUMN,  name,
+		                                  CHAT_USERS_FLAGS_COLUMN, flags,
+		                                  CHAT_USERS_COLOR_COLUMN, &send_color,
+		                                  CHAT_USERS_WEIGHT_COLUMN, is_buddy ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL,
+		                                  -1);
+	}
+	else
+	{
+		gtk_list_store_insert_with_values(ls, &iter,
+		                                  -1, /* "row" */
+		                                  CHAT_USERS_ICON_COLUMN,  pixbuf,
+		                                  CHAT_USERS_ALIAS_COLUMN, alias,
+		                                  CHAT_USERS_ALIAS_KEY_COLUMN, alias_key,
+		                                  CHAT_USERS_NAME_COLUMN,  name,
+		                                  CHAT_USERS_FLAGS_COLUMN, flags,
+		                                  CHAT_USERS_COLOR_COLUMN, get_nick_color(gtkconv, name),
+		                                  CHAT_USERS_WEIGHT_COLUMN, is_buddy ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL,
+		                                  -1);
+#else
+		gtk_list_store_append(ls, &iter);
+		gtk_list_store_set(ls, &iter,
+		                   CHAT_USERS_ICON_COLUMN,  pixbuf,
+		                   CHAT_USERS_ALIAS_COLUMN, alias,
+		                   CHAT_USERS_ALIAS_KEY_COLUMN, alias_key,
+		                   CHAT_USERS_NAME_COLUMN,  name,
+		                   CHAT_USERS_FLAGS_COLUMN, flags,
+		                   CHAT_USERS_COLOR_COLUMN, &send_color,
+		                   CHAT_USERS_WEIGHT_COLUMN, is_buddy ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL,
+		                   -1);
+	}
+	else
+	{
+		gtk_list_store_append(ls, &iter);
+		gtk_list_store_set(ls, &iter,
+		                   CHAT_USERS_ICON_COLUMN,  pixbuf,
+		                   CHAT_USERS_ALIAS_COLUMN, alias,
+		                   CHAT_USERS_ALIAS_KEY_COLUMN, alias_key,
+		                   CHAT_USERS_NAME_COLUMN,  name,
+		                   CHAT_USERS_FLAGS_COLUMN, flags,
+		                   CHAT_USERS_COLOR_COLUMN, get_nick_color(gtkconv, name),
+		                   CHAT_USERS_WEIGHT_COLUMN, is_buddy ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL,
+		                   -1);
+#endif
+	}
+
+	if (pixbuf)
+		g_object_unref(pixbuf);
+	g_free(alias_key);
+}
+
+static void
+tab_complete_process_item(int *most_matched, char *entered, char **partial, char *nick_partial,
+				  GList **matches, gboolean command, char *name)
+{
+	strncpy(nick_partial, name, strlen(entered));
+	nick_partial[strlen(entered)] = '\0';
+	if (gaim_utf8_strcasecmp(nick_partial, entered))
+		return;
+
+	/* if we're here, it's a possible completion */
+
+	if (*most_matched == -1) {
+		/*
+		 * this will only get called once, since from now
+		 * on *most_matched is >= 0
+		 */
+		*most_matched = strlen(name);
+		*partial = g_strdup(name);
+	}
+	else if (*most_matched) {
+		char *tmp = g_strdup(name);
+
+		while (gaim_utf8_strcasecmp(tmp, *partial)) {
+			(*partial)[*most_matched] = '\0';
+			if (*most_matched < strlen(tmp))
+				tmp[*most_matched] = '\0';
+			(*most_matched)--;
+		}
+		(*most_matched)++;
+
+		g_free(tmp);
+	}
+
+	*matches = g_list_insert_sorted(*matches, g_strdup(name),
+								   (GCompareFunc)gaim_utf8_strcasecmp);
+}
+
+static gboolean
+tab_complete(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv;
+	GtkTextIter cursor, word_start, start_buffer;
+	int start;
+	int most_matched = -1;
+	char *entered, *partial = NULL;
+	char *text;
+	char *nick_partial;
+	const char *prefix;
+	GList *matches = NULL;
+	gboolean command = FALSE;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtk_text_buffer_get_start_iter(gtkconv->entry_buffer, &start_buffer);
+	gtk_text_buffer_get_iter_at_mark(gtkconv->entry_buffer, &cursor,
+			gtk_text_buffer_get_insert(gtkconv->entry_buffer));
+
+	word_start = cursor;
+
+	/* if there's nothing there just return */
+	if (!gtk_text_iter_compare(&cursor, &start_buffer))
+		return (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) ? TRUE : FALSE;
+
+	text = gtk_text_buffer_get_text(gtkconv->entry_buffer, &start_buffer,
+									&cursor, FALSE);
+
+	/* if we're at the end of ": " we need to move back 2 spaces */
+	start = strlen(text) - 1;
+
+	if (strlen(text) >= 2 && !strncmp(&text[start-1], ": ", 2)) {
+		gtk_text_iter_backward_chars(&word_start, 2);
+		start-=2;
+	}
+
+	/* find the start of the word that we're tabbing */
+	while (start >= 0 && text[start] != ' ') {
+		gtk_text_iter_backward_char(&word_start);
+		start--;
+	}
+
+	prefix = gaim_gtk_get_cmd_prefix();
+	if (start == -1 && (strlen(text) >= strlen(prefix)) && !strncmp(text, prefix, strlen(prefix))) {
+		command = TRUE;
+		gtk_text_iter_forward_chars(&word_start, strlen(prefix));
+	}
+
+	g_free(text);
+
+	entered = gtk_text_buffer_get_text(gtkconv->entry_buffer, &word_start,
+									   &cursor, FALSE);
+
+	if (!g_utf8_strlen(entered, -1)) {
+		g_free(entered);
+		return (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) ? TRUE : FALSE;
+	}
+
+	nick_partial = g_malloc(strlen(entered)+1);
+
+	if (command) {
+		GList *list = gaim_cmd_list(conv);
+		GList *l;
+
+		/* Commands */
+		for (l = list; l != NULL; l = l->next) {
+			tab_complete_process_item(&most_matched, entered, &partial, nick_partial,
+									  &matches, TRUE, l->data);
+		}
+		g_list_free(list);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		GaimConvChat *chat = GAIM_CONV_CHAT(conv);
+		GList *l = gaim_conv_chat_get_users(chat);
+		GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(GAIM_GTK_CONVERSATION(conv)->u.chat->list));
+		GtkTreeIter iter;
+		int f;
+
+		/* Users */
+		for (; l != NULL; l = l->next) {
+			tab_complete_process_item(&most_matched, entered, &partial, nick_partial,
+									  &matches, TRUE, ((GaimConvChatBuddy *)l->data)->name);
+		}
+
+
+		/* Aliases */
+		if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+		{
+			do {
+				char *name;
+				char *alias;
+
+				gtk_tree_model_get(model, &iter,
+						   CHAT_USERS_NAME_COLUMN, &name,
+						   CHAT_USERS_ALIAS_COLUMN, &alias,
+						   -1);
+
+				if (strcmp(name, alias))
+					tab_complete_process_item(&most_matched, entered, &partial, nick_partial,
+										  &matches, FALSE, alias);
+				g_free(name);
+				g_free(alias);
+
+				f = gtk_tree_model_iter_next(model, &iter);
+			} while (f != 0);
+		}
+	} else {
+		g_free(nick_partial);
+		g_free(entered);
+		return FALSE;
+	}
+
+	g_free(nick_partial);
+
+	/* we're only here if we're doing new style */
+
+	/* if there weren't any matches, return */
+	if (!matches) {
+		/* if matches isn't set partials won't be either */
+		g_free(entered);
+		return (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) ? TRUE : FALSE;
+	}
+
+	gtk_text_buffer_delete(gtkconv->entry_buffer, &word_start, &cursor);
+
+	if (!matches->next) {
+		/* there was only one match. fill it in. */
+		gtk_text_buffer_get_start_iter(gtkconv->entry_buffer, &start_buffer);
+		gtk_text_buffer_get_iter_at_mark(gtkconv->entry_buffer, &cursor,
+				gtk_text_buffer_get_insert(gtkconv->entry_buffer));
+
+		if (!gtk_text_iter_compare(&cursor, &start_buffer)) {
+			char *tmp = g_strdup_printf("%s: ", (char *)matches->data);
+			gtk_text_buffer_insert_at_cursor(gtkconv->entry_buffer, tmp, -1);
+			g_free(tmp);
+		}
+		else
+			gtk_text_buffer_insert_at_cursor(gtkconv->entry_buffer,
+											 matches->data, -1);
+
+		g_free(matches->data);
+		matches = g_list_remove(matches, matches->data);
+	}
+	else {
+		/*
+		 * there were lots of matches, fill in as much as possible
+		 * and display all of them
+		 */
+		char *addthis = g_malloc0(1);
+
+		while (matches) {
+			char *tmp = addthis;
+			addthis = g_strconcat(tmp, matches->data, " ", NULL);
+			g_free(tmp);
+			g_free(matches->data);
+			matches = g_list_remove(matches, matches->data);
+		}
+
+		gaim_conversation_write(conv, NULL, addthis, GAIM_MESSAGE_NO_LOG,
+								time(NULL));
+		gtk_text_buffer_insert_at_cursor(gtkconv->entry_buffer, partial, -1);
+		g_free(addthis);
+	}
+
+	g_free(entered);
+	g_free(partial);
+
+	return TRUE;
+}
+
+static void topic_callback(GtkWidget *w, GaimGtkConversation *gtkconv)
+{
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	GaimConnection *gc;
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkChatPane *gtkchat;
+	char *new_topic;
+	const char *current_topic;
+
+	gc      = gaim_conversation_get_gc(conv);
+
+	if(!gc || !(prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl)))
+		return;
+
+	if(prpl_info->set_chat_topic == NULL)
+		return;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+	new_topic = g_strdup(gtk_entry_get_text(GTK_ENTRY(gtkchat->topic_text)));
+	current_topic = gaim_conv_chat_get_topic(GAIM_CONV_CHAT(conv));
+
+	if(current_topic && !g_utf8_collate(new_topic, current_topic)){
+		g_free(new_topic);
+		return;
+	}
+
+	gtk_entry_set_text(GTK_ENTRY(gtkchat->topic_text), current_topic);
+
+	prpl_info->set_chat_topic(gc, gaim_conv_chat_get_id(GAIM_CONV_CHAT(conv)),
+			new_topic);
+
+	g_free(new_topic);
+}
+
+static gint
+sort_chat_users(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer userdata)
+{
+	GaimConvChatBuddyFlags f1 = 0, f2 = 0;
+	char *user1 = NULL, *user2 = NULL;
+	gboolean buddy1 = FALSE, buddy2 = FALSE;
+	gint ret = 0;
+
+	gtk_tree_model_get(model, a,
+	                   CHAT_USERS_ALIAS_KEY_COLUMN, &user1,
+	                   CHAT_USERS_FLAGS_COLUMN, &f1,
+	                   CHAT_USERS_WEIGHT_COLUMN, &buddy1,
+	                   -1);
+	gtk_tree_model_get(model, b,
+	                   CHAT_USERS_ALIAS_KEY_COLUMN, &user2,
+	                   CHAT_USERS_FLAGS_COLUMN, &f2,
+	                   CHAT_USERS_WEIGHT_COLUMN, &buddy2,
+	                   -1);
+
+	if (user1 == NULL || user2 == NULL) {
+		if (!(user1 == NULL && user2 == NULL))
+			ret = (user1 == NULL) ? -1: 1;
+	} else if (f1 != f2) {
+		/* sort more important users first */
+		ret = (f1 > f2) ? -1 : 1;
+	} else if (buddy1 != buddy2) {
+		ret = (buddy1 > buddy2) ? -1 : 1;
+	} else {
+		ret = strcmp(user1, user2);
+	}
+
+	g_free(user1);
+	g_free(user2);
+
+	return ret;
+}
+
+static void
+update_chat_alias(GaimBuddy *buddy, GaimConversation *conv, GaimConnection *gc, GaimPluginProtocolInfo *prpl_info)
+{
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+	GaimConvChat *chat = GAIM_CONV_CHAT(conv);
+	GtkTreeModel *model;
+	char *normalized_name;
+	GtkTreeIter iter;
+	int f;
+
+	g_return_if_fail(buddy != NULL);
+	g_return_if_fail(conv != NULL);
+
+	/* This is safe because this callback is only used in chats, not IMs. */
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkconv->u.chat->list));
+
+	if (!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+		return;
+
+	normalized_name = g_strdup(gaim_normalize(conv->account, buddy->name));
+
+	do {
+		char *name;
+
+		gtk_tree_model_get(model, &iter, CHAT_USERS_NAME_COLUMN, &name, -1);
+
+		if (!strcmp(normalized_name, gaim_normalize(conv->account, name))) {
+			const char *alias = name;
+			char *tmp;
+			char *alias_key = NULL;
+			GaimBuddy *buddy2;
+
+			if (strcmp(chat->nick, gaim_normalize(conv->account, name))) {
+				/* This user is not me, so look into updating the alias. */
+
+				if ((buddy2 = gaim_find_buddy(conv->account, name)) != NULL) {
+					alias = gaim_buddy_get_contact_alias(buddy2);
+				}
+
+				tmp = g_utf8_casefold(alias, -1);
+				alias_key = g_utf8_collate_key(tmp, -1);
+				g_free(tmp);
+
+				gtk_list_store_set(GTK_LIST_STORE(model), &iter,
+								CHAT_USERS_ALIAS_COLUMN, alias,
+								CHAT_USERS_ALIAS_KEY_COLUMN, alias_key,
+								-1);
+				g_free(alias_key);
+			}
+			g_free(name);
+			break;
+		}
+
+		f = gtk_tree_model_iter_next(model, &iter);
+
+		g_free(name);
+	} while (f != 0);
+
+	g_free(normalized_name);
+}
+
+static void
+blist_node_aliased_cb(GaimBlistNode *node, const char *old_alias, GaimConversation *conv)
+{
+	GaimConnection *gc;
+	GaimPluginProtocolInfo *prpl_info;
+
+	g_return_if_fail(node != NULL);
+	g_return_if_fail(conv != NULL);
+
+	gc = gaim_conversation_get_gc(conv);
+	g_return_if_fail(gc != NULL);
+	g_return_if_fail(gc->prpl != NULL);
+	prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	if (prpl_info->options & OPT_PROTO_UNIQUE_CHATNAME)
+		return;
+
+	if (GAIM_BLIST_NODE_IS_CONTACT(node))
+	{
+		GaimBlistNode *bnode;
+
+		for(bnode = node->child; bnode; bnode = bnode->next) {
+
+			if(!GAIM_BLIST_NODE_IS_BUDDY(bnode))
+				continue;
+
+			update_chat_alias((GaimBuddy *)bnode, conv, gc, prpl_info);
+		}
+	}
+	else if (GAIM_BLIST_NODE_IS_BUDDY(node))
+		update_chat_alias((GaimBuddy *)node, conv, gc, prpl_info);
+}
+
+static void
+buddy_cb_common(GaimBuddy *buddy, GaimConversation *conv, gboolean is_buddy)
+{
+	GtkTreeModel *model;
+	char *normalized_name;
+	GtkTreeIter iter;
+	int f;
+
+	g_return_if_fail(buddy != NULL);
+	g_return_if_fail(conv != NULL);
+
+	/* Do nothing if the buddy does not belong to the conv's account */
+	if (gaim_buddy_get_account(buddy) != gaim_conversation_get_account(conv))
+		return;
+
+	/* This is safe because this callback is only used in chats, not IMs. */
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(GAIM_GTK_CONVERSATION(conv)->u.chat->list));
+
+	if (!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+		return;
+
+	normalized_name = g_strdup(gaim_normalize(conv->account, buddy->name));
+
+	do {
+		char *name;
+
+		gtk_tree_model_get(model, &iter, CHAT_USERS_NAME_COLUMN, &name, -1);
+
+		if (!strcmp(normalized_name, gaim_normalize(conv->account, name))) {
+			gtk_list_store_set(GTK_LIST_STORE(model), &iter,
+			                   CHAT_USERS_WEIGHT_COLUMN, is_buddy ? PANGO_WEIGHT_BOLD : PANGO_WEIGHT_NORMAL, -1);
+			g_free(name);
+			break;
+		}
+
+		f = gtk_tree_model_iter_next(model, &iter);
+
+		g_free(name);
+	} while (f != 0);
+
+	g_free(normalized_name);
+
+	blist_node_aliased_cb((GaimBlistNode *)buddy, NULL, conv);
+}
+
+static void
+buddy_added_cb(GaimBuddy *buddy, GaimConversation *conv)
+{
+	buddy_cb_common(buddy, conv, TRUE);
+}
+
+static void
+buddy_removed_cb(GaimBuddy *buddy, GaimConversation *conv)
+{
+	/* If there's another buddy for the same "dude" on the list, do nothing. */
+	if (gaim_find_buddy(buddy->account, buddy->name) != NULL)
+		return;
+
+	buddy_cb_common(buddy, conv, FALSE);
+}
+
+static void send_menu_cb(GtkWidget *widget, GaimGtkConversation *gtkconv)
+{
+	g_signal_emit_by_name(gtkconv->entry, "message_send");
+}
+
+static void
+entry_popup_menu_cb(GtkIMHtml *imhtml, GtkMenu *menu, gpointer data)
+{
+	GtkWidget *menuitem;
+	GaimGtkConversation *gtkconv = data;
+
+	g_return_if_fail(menu != NULL);
+	g_return_if_fail(gtkconv != NULL);
+
+	menuitem = gaim_new_item_from_stock(NULL, _("_Send"), GAIM_STOCK_SEND,
+										G_CALLBACK(send_menu_cb), gtkconv,
+										0, 0, NULL);
+	if (gtk_text_buffer_get_char_count(imhtml->text_buffer) == 0)
+		gtk_widget_set_sensitive(menuitem, FALSE);
+	gtk_menu_shell_insert(GTK_MENU_SHELL(menu), menuitem, 0);
+
+	menuitem = gtk_separator_menu_item_new();
+	gtk_widget_show(menuitem);
+	gtk_menu_shell_insert(GTK_MENU_SHELL(menu), menuitem, 1);
+}
+
+
+static void resize_imhtml_cb(GaimGtkConversation *gtkconv)
+{
+	GtkTextBuffer *buffer;
+	GtkTextIter iter;
+        int wrapped_lines;
+        int lines;
+        GdkRectangle oneline;
+	GtkRequisition sr;
+        int height;
+        int pad_top, pad_inside, pad_bottom;
+
+	buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->entry));
+
+        wrapped_lines = 1;
+        gtk_text_buffer_get_start_iter(buffer, &iter);
+        gtk_text_view_get_iter_location(GTK_TEXT_VIEW(gtkconv->entry), &iter, &oneline);
+        while (gtk_text_view_forward_display_line(GTK_TEXT_VIEW(gtkconv->entry), &iter))
+                wrapped_lines++;
+
+        lines = gtk_text_buffer_get_line_count(buffer);
+
+        /* Show a maximum of 4 lines */
+        lines = MIN(lines, 4);
+        wrapped_lines = MIN(wrapped_lines, 4);
+
+        pad_top = gtk_text_view_get_pixels_above_lines(GTK_TEXT_VIEW(gtkconv->entry));
+        pad_bottom = gtk_text_view_get_pixels_below_lines(GTK_TEXT_VIEW(gtkconv->entry));
+        pad_inside = gtk_text_view_get_pixels_inside_wrap(GTK_TEXT_VIEW(gtkconv->entry));
+
+        height = (oneline.height + pad_top + pad_bottom) * lines;
+        height += (oneline.height + pad_inside) * (wrapped_lines - lines);
+
+	gtk_widget_size_request(gtkconv->lower_hbox, &sr);
+	if (sr.height < height + GAIM_HIG_BOX_SPACE) {
+		gtkconv->auto_resize = TRUE;
+		gtkconv->entry_growing = TRUE;
+	        gtk_widget_set_size_request(gtkconv->lower_hbox, -1, height + GAIM_HIG_BOX_SPACE);
+	        g_idle_add(reset_auto_resize_cb, gtkconv);
+	}
+}
+
+static GtkWidget *
+setup_chat_pane(GaimGtkConversation *gtkconv)
+{
+	GaimPluginProtocolInfo *prpl_info;
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkChatPane *gtkchat;
+	GaimConnection *gc;
+	GtkWidget *vpaned, *hpaned;
+	GtkWidget *vbox, *hbox, *frame;
+	GtkWidget *imhtml_sw;
+	GtkPolicyType imhtml_sw_hscroll;
+	GtkWidget *lbox;
+	GtkWidget *label;
+	GtkWidget *list;
+	GtkWidget *sw;
+	GtkListStore *ls;
+	GtkCellRenderer *rend;
+	GtkTreeViewColumn *col;
+	void *blist_handle = gaim_blist_get_handle();
+	GList *focus_chain = NULL;
+
+	gtkchat = gtkconv->u.chat;
+	gc      = gaim_conversation_get_gc(conv);
+	g_return_val_if_fail(gc != NULL, NULL);
+	g_return_val_if_fail(gc->prpl != NULL, NULL);
+	prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	/* Setup the outer pane. */
+	vpaned = gtk_vpaned_new();
+	gtk_widget_show(vpaned);
+
+	/* Setup the top part of the pane. */
+	vbox = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_paned_pack1(GTK_PANED(vpaned), vbox, TRUE, TRUE);
+	gtk_widget_show(vbox);
+
+	if (prpl_info->options & OPT_PROTO_CHAT_TOPIC)
+	{
+		hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+		gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
+		gtk_widget_show(hbox);
+
+		label = gtk_label_new(_("Topic:"));
+		gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
+		gtk_widget_show(label);
+
+		gtkchat->topic_text = gtk_entry_new();
+
+		if(prpl_info->set_chat_topic == NULL) {
+			gtk_editable_set_editable(GTK_EDITABLE(gtkchat->topic_text), FALSE);
+		} else {
+			g_signal_connect(GTK_OBJECT(gtkchat->topic_text), "activate",
+					G_CALLBACK(topic_callback), gtkconv);
+		}
+
+		gtk_box_pack_start(GTK_BOX(hbox), gtkchat->topic_text, TRUE, TRUE, 0);
+		gtk_widget_show(gtkchat->topic_text);
+	}
+
+	/* Setup the horizontal pane. */
+	hpaned = gtk_hpaned_new();
+	gtk_box_pack_start(GTK_BOX(vbox), hpaned, TRUE, TRUE, 0);
+	gtk_widget_show(hpaned);
+
+	/* Setup gtkihmtml. */
+	frame = gaim_gtk_create_imhtml(FALSE, &gtkconv->imhtml, NULL, &imhtml_sw);
+	gtk_widget_set_name(gtkconv->imhtml, "gaim_gtkconv_imhtml");
+	gtk_imhtml_show_comments(GTK_IMHTML(gtkconv->imhtml), TRUE);
+	gtk_paned_pack1(GTK_PANED(hpaned), frame, TRUE, TRUE);
+	gtk_widget_show(frame);
+	gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(imhtml_sw),
+	                               &imhtml_sw_hscroll, NULL);
+	gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(imhtml_sw),
+	                               imhtml_sw_hscroll, GTK_POLICY_ALWAYS);
+
+	gtk_widget_set_size_request(gtkconv->imhtml,
+			gaim_prefs_get_int("/gaim/gtk/conversations/chat/default_width"),
+			gaim_prefs_get_int("/gaim/gtk/conversations/chat/default_height"));
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "size-allocate",
+					 G_CALLBACK(size_allocate_cb), gtkconv);
+
+	g_signal_connect_after(G_OBJECT(gtkconv->imhtml), "button_press_event",
+						   G_CALLBACK(entry_stop_rclick_cb), NULL);
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "key_press_event",
+						   G_CALLBACK(refocus_entry_cb), gtkconv);
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "key_release_event",
+						   G_CALLBACK(refocus_entry_cb), gtkconv);
+
+	/* Build the right pane. */
+	lbox = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_paned_pack2(GTK_PANED(hpaned), lbox, FALSE, TRUE);
+	gtk_widget_show(lbox);
+
+	/* Setup the label telling how many people are in the room. */
+	gtkchat->count = gtk_label_new(_("0 people in room"));
+	gtk_box_pack_start(GTK_BOX(lbox), gtkchat->count, FALSE, FALSE, 0);
+	gtk_widget_show(gtkchat->count);
+
+	/* Setup the list of users. */
+	sw = gtk_scrolled_window_new(NULL, NULL);
+	gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw),
+								   GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
+	gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
+	gtk_box_pack_start(GTK_BOX(lbox), sw, TRUE, TRUE, 0);
+	gtk_widget_show(sw);
+
+	ls = gtk_list_store_new(CHAT_USERS_COLUMNS, GDK_TYPE_PIXBUF, G_TYPE_STRING,
+							G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT,
+							GDK_TYPE_COLOR, G_TYPE_INT);
+	gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(ls), CHAT_USERS_ALIAS_KEY_COLUMN,
+									sort_chat_users, NULL, NULL);
+
+	list = gtk_tree_view_new_with_model(GTK_TREE_MODEL(ls));
+
+	rend = gtk_cell_renderer_pixbuf_new();
+
+	col = gtk_tree_view_column_new_with_attributes(NULL, rend,
+												   "pixbuf", CHAT_USERS_ICON_COLUMN, NULL);
+	gtk_tree_view_column_set_sizing(col, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
+	gtk_tree_view_append_column(GTK_TREE_VIEW(list), col);
+	gtk_widget_set_size_request(lbox,
+	                            gaim_prefs_get_int("/gaim/gtk/conversations/chat/userlist_width"), -1);
+
+	g_signal_connect(G_OBJECT(list), "button_press_event",
+					 G_CALLBACK(right_click_chat_cb), gtkconv);
+	g_signal_connect(G_OBJECT(list), "popup-menu",
+			 G_CALLBACK(gtkconv_chat_popup_menu_cb), gtkconv);
+	g_signal_connect(G_OBJECT(lbox), "size-allocate", G_CALLBACK(lbox_size_allocate_cb), gtkconv);
+
+
+	rend = gtk_cell_renderer_text_new();
+
+	g_object_set(rend,
+				 "foreground-set", TRUE,
+				 "weight-set", TRUE,
+				 NULL);
+	col = gtk_tree_view_column_new_with_attributes(NULL, rend,
+	                                               "text", CHAT_USERS_ALIAS_COLUMN,
+	                                               "foreground-gdk", CHAT_USERS_COLOR_COLUMN,
+	                                               "weight", CHAT_USERS_WEIGHT_COLUMN,
+	                                               NULL);
+
+	gaim_signal_connect(blist_handle, "buddy-added",
+						gtkchat, GAIM_CALLBACK(buddy_added_cb), conv);
+	gaim_signal_connect(blist_handle, "buddy-removed",
+						gtkchat, GAIM_CALLBACK(buddy_removed_cb), conv);
+	gaim_signal_connect(blist_handle, "blist-node-aliased",
+						gtkchat, GAIM_CALLBACK(blist_node_aliased_cb), conv);
+
+#if GTK_CHECK_VERSION(2,6,0)
+	gtk_tree_view_column_set_expand(col, TRUE);
+	g_object_set(rend, "ellipsize", PANGO_ELLIPSIZE_END, NULL);
+#endif
+
+	gtk_tree_view_append_column(GTK_TREE_VIEW(list), col);
+
+	gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(list), FALSE);
+	gtk_widget_show(list);
+
+	gtkchat->list = list;
+
+	gtk_container_add(GTK_CONTAINER(sw), list);
+
+	/* Setup the bottom half of the conversation window */
+	vbox = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_paned_pack2(GTK_PANED(vpaned), vbox, FALSE, TRUE);
+	gtk_widget_show(vbox);
+
+	gtkconv->lower_hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_box_pack_start(GTK_BOX(vbox), gtkconv->lower_hbox, TRUE, TRUE, 0);
+	gtk_widget_show(gtkconv->lower_hbox);
+
+	vbox = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_box_pack_end(GTK_BOX(gtkconv->lower_hbox), vbox, TRUE, TRUE, 0);
+	gtk_widget_show(vbox);
+
+	/* Setup the toolbar, entry widget and all signals */
+	frame = gaim_gtk_create_imhtml(TRUE, &gtkconv->entry, &gtkconv->toolbar, NULL);
+	gtk_box_pack_start(GTK_BOX(vbox), frame, TRUE, TRUE, 0);
+	gtk_widget_show(frame);
+
+	g_signal_connect(G_OBJECT(gtkconv->entry), "populate-popup",
+					 G_CALLBACK(entry_popup_menu_cb), gtkconv);
+
+	gtk_widget_set_name(gtkconv->entry, "gaim_gtkconv_entry");
+	gtk_imhtml_set_protocol_name(GTK_IMHTML(gtkconv->entry),
+								 gaim_account_get_protocol_name(conv->account));
+	gtk_widget_set_size_request(gtkconv->lower_hbox, -1,
+			gaim_prefs_get_int("/gaim/gtk/conversations/chat/entry_height"));
+	gtkconv->entry_buffer =
+		gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->entry));
+	g_object_set_data(G_OBJECT(gtkconv->entry_buffer), "user_data", gtkconv);
+	g_signal_connect_swapped(G_OBJECT(gtkconv->entry_buffer), "changed",
+                                 G_CALLBACK(resize_imhtml_cb), gtkconv);
+	
+	g_signal_connect(G_OBJECT(gtkconv->entry), "key_press_event",
+	                 G_CALLBACK(entry_key_press_cb), gtkconv);
+	g_signal_connect_after(G_OBJECT(gtkconv->entry), "message_send",
+	                       G_CALLBACK(send_cb), gtkconv);
+	g_signal_connect_after(G_OBJECT(gtkconv->entry), "button_press_event",
+	                       G_CALLBACK(entry_stop_rclick_cb), NULL);
+	g_signal_connect(G_OBJECT(gtkconv->lower_hbox), "size-allocate",
+	                 G_CALLBACK(size_allocate_cb), gtkconv);
+
+	default_formatize(gtkconv);
+
+	/*
+	 * Focus for chat windows should be as follows:
+	 * Tab title -> chat topic -> conversation scrollback -> user list ->
+	 *   user list buttons -> entry -> buttons at bottom
+	 */
+	focus_chain = g_list_prepend(focus_chain, gtkconv->entry);
+	gtk_container_set_focus_chain(GTK_CONTAINER(vbox), focus_chain);
+
+	return vpaned;
+}
+
+static GtkWidget *
+setup_im_pane(GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GtkWidget *frame;
+	GtkWidget *imhtml_sw;
+	GtkPolicyType imhtml_sw_hscroll;
+	GtkWidget *paned;
+	GtkWidget *vbox;
+	GtkWidget *vbox2;
+	GList *focus_chain = NULL;
+
+	/* Setup the outer pane */
+	paned = gtk_vpaned_new();
+	gtk_widget_show(paned);
+
+	/* Setup the top part of the pane */
+	vbox = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_paned_pack1(GTK_PANED(paned), vbox, TRUE, TRUE);
+	gtk_widget_show(vbox);
+
+	/* Setup the gtkimhtml widget */
+	frame = gaim_gtk_create_imhtml(FALSE, &gtkconv->imhtml, NULL, &imhtml_sw);
+	gtk_widget_set_name(gtkconv->imhtml, "gaim_gtkconv_imhtml");
+	gtk_imhtml_show_comments(GTK_IMHTML(gtkconv->imhtml),TRUE);
+	gtk_box_pack_start(GTK_BOX(vbox), frame, TRUE, TRUE, 0);
+	gtk_widget_show(frame);
+	gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(imhtml_sw),
+	                               &imhtml_sw_hscroll, NULL);
+	gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(imhtml_sw),
+	                               imhtml_sw_hscroll, GTK_POLICY_ALWAYS);
+
+	gtk_widget_set_size_request(gtkconv->imhtml,
+			gaim_prefs_get_int("/gaim/gtk/conversations/im/default_width"),
+			gaim_prefs_get_int("/gaim/gtk/conversations/im/default_height"));
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "size-allocate",
+	                 G_CALLBACK(size_allocate_cb), gtkconv);
+
+	g_signal_connect_after(G_OBJECT(gtkconv->imhtml), "button_press_event",
+	                       G_CALLBACK(entry_stop_rclick_cb), NULL);
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "key_press_event",
+	                 G_CALLBACK(refocus_entry_cb), gtkconv);
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "key_release_event",
+	                 G_CALLBACK(refocus_entry_cb), gtkconv);
+
+	/* Setup the bottom half of the conversation window */
+	vbox2 = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_paned_pack2(GTK_PANED(paned), vbox2, FALSE, TRUE);
+	gtk_widget_show(vbox2);
+
+	gtkconv->lower_hbox = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_box_pack_start(GTK_BOX(vbox2), gtkconv->lower_hbox, TRUE, TRUE, 0);
+	gtk_widget_show(gtkconv->lower_hbox);
+
+	vbox2 = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtk_box_pack_end(GTK_BOX(gtkconv->lower_hbox), vbox2, TRUE, TRUE, 0);
+	gtk_widget_show(vbox2);
+
+	/* Setup the toolbar, entry widget and all signals */
+	frame = gaim_gtk_create_imhtml(TRUE, &gtkconv->entry, &gtkconv->toolbar, NULL);
+	gtk_box_pack_start(GTK_BOX(vbox2), frame, TRUE, TRUE, 0);
+	gtk_widget_show(frame);
+
+	g_signal_connect(G_OBJECT(gtkconv->entry), "populate-popup",
+					 G_CALLBACK(entry_popup_menu_cb), gtkconv);
+
+	gtk_widget_set_name(gtkconv->entry, "gaim_gtkconv_entry");
+	gtk_imhtml_set_protocol_name(GTK_IMHTML(gtkconv->entry),
+								 gaim_account_get_protocol_name(conv->account));
+	gtk_widget_set_size_request(gtkconv->lower_hbox, -1,
+			gaim_prefs_get_int("/gaim/gtk/conversations/im/entry_height"));
+	gtkconv->entry_buffer =
+		gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->entry));
+	g_object_set_data(G_OBJECT(gtkconv->entry_buffer), "user_data", gtkconv);
+
+	g_signal_connect(G_OBJECT(gtkconv->entry), "key_press_event",
+	                 G_CALLBACK(entry_key_press_cb), gtkconv);
+	g_signal_connect_after(G_OBJECT(gtkconv->entry), "message_send",
+	                       G_CALLBACK(send_cb), gtkconv);
+	g_signal_connect_after(G_OBJECT(gtkconv->entry), "button_press_event",
+	                       G_CALLBACK(entry_stop_rclick_cb), NULL);
+	g_signal_connect(G_OBJECT(gtkconv->lower_hbox), "size-allocate",
+	                 G_CALLBACK(size_allocate_cb), gtkconv);
+
+	g_signal_connect(G_OBJECT(gtkconv->entry_buffer), "insert_text",
+	                 G_CALLBACK(insert_text_cb), gtkconv);
+	g_signal_connect(G_OBJECT(gtkconv->entry_buffer), "delete_range",
+	                 G_CALLBACK(delete_text_cb), gtkconv);
+	g_signal_connect_swapped(G_OBJECT(gtkconv->entry_buffer), "changed",
+				 G_CALLBACK(resize_imhtml_cb), gtkconv);
+
+	/* had to move this after the imtoolbar is attached so that the
+	 * signals get fired to toggle the buttons on the toolbar as well.
+	 */
+	default_formatize(gtkconv);
+
+	g_signal_connect_after(G_OBJECT(gtkconv->entry), "format_function_clear",
+						   G_CALLBACK(clear_formatting_cb), gtkconv);
+
+	gtkconv->u.im->animate = gaim_prefs_get_bool("/gaim/gtk/conversations/im/animate_buddy_icons");
+	gtkconv->u.im->show_icon = TRUE;
+
+	/*
+	 * Focus for IM windows should be as follows:
+	 * Tab title -> conversation scrollback -> entry
+	 */
+	focus_chain = g_list_prepend(focus_chain, gtkconv->entry);
+	gtk_container_set_focus_chain(GTK_CONTAINER(vbox2), focus_chain);
+
+	return paned;
+}
+
+static void
+conv_dnd_recv(GtkWidget *widget, GdkDragContext *dc, guint x, guint y,
+              GtkSelectionData *sd, guint info, guint t,
+              GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkWindow *win = gtkconv->win;
+	GaimConversation *c;
+	if (sd->target == gdk_atom_intern("GAIM_BLIST_NODE", FALSE))
+	{
+		GaimBlistNode *n = NULL;
+		GaimBuddy *b;
+		GaimGtkConversation *gtkconv = NULL;
+
+		n = *(GaimBlistNode **)sd->data;
+
+		if (GAIM_BLIST_NODE_IS_CONTACT(n))
+			b = gaim_contact_get_priority_buddy((GaimContact*)n);
+		else if (GAIM_BLIST_NODE_IS_BUDDY(n))
+			b = (GaimBuddy*)n;
+		else
+			return;
+
+		/*
+		 * If we already have an open conversation with this buddy, then
+		 * just move the conv to this window.  Otherwise, create a new
+		 * conv and add it to this window.
+		 */
+		c = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, b->name, b->account);
+		if (c != NULL) {
+			GaimGtkWindow *oldwin;
+			gtkconv = GAIM_GTK_CONVERSATION(c);
+			oldwin = gtkconv->win;
+			if (oldwin != win) {
+				gaim_gtk_conv_window_remove_gtkconv(oldwin, gtkconv);
+				gaim_gtk_conv_window_add_gtkconv(win, gtkconv);
+			}
+		} else {
+			c = gaim_conversation_new(GAIM_CONV_TYPE_IM, b->account, b->name);
+			gtkconv = GAIM_GTK_CONVERSATION(c);
+			if (gtkconv->win != win)
+			{
+				gaim_gtk_conv_window_remove_gtkconv(gtkconv->win, gtkconv);
+				gaim_gtk_conv_window_add_gtkconv(win, gtkconv);
+			}
+		}
+
+		/* Make this conversation the active conversation */
+		gaim_gtk_conv_window_switch_gtkconv(win, gtkconv);
+
+		gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
+	}
+	else if (sd->target == gdk_atom_intern("application/x-im-contact", FALSE))
+	{
+		char *protocol = NULL;
+		char *username = NULL;
+		GaimAccount *account;
+		GaimGtkConversation *gtkconv;
+
+		if (gaim_gtk_parse_x_im_contact((const char *)sd->data, FALSE, &account,
+						&protocol, &username, NULL))
+		{
+			if (account == NULL)
+			{
+				gaim_notify_error(win, NULL,
+					_("You are not currently signed on with an account that "
+					  "can add that buddy."), NULL);
+			}
+			else
+			{
+				c = gaim_conversation_new(GAIM_CONV_TYPE_IM, account, username);
+				gtkconv = GAIM_GTK_CONVERSATION(c);
+				if (gtkconv->win != win)
+				{
+					gaim_gtk_conv_window_remove_gtkconv(gtkconv->win, gtkconv);
+					gaim_gtk_conv_window_add_gtkconv(win, gtkconv);
+				}
+			}
+		}
+
+		g_free(username);
+		g_free(protocol);
+
+		gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
+	}
+	else if (sd->target == gdk_atom_intern("text/uri-list", FALSE)) {
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+			gaim_dnd_file_manage(sd, gaim_conversation_get_account(conv), gaim_conversation_get_name(conv));
+		gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
+	}
+	else
+		gtk_drag_finish(dc, FALSE, FALSE, t);
+}
+
+
+static const GtkTargetEntry te[] =
+{
+	GTK_IMHTML_DND_TARGETS,
+	{"GAIM_BLIST_NODE", GTK_TARGET_SAME_APP, GTK_IMHTML_DRAG_NUM},
+	{"application/x-im-contact", 0, GTK_IMHTML_DRAG_NUM + 1}
+};
+
+static GaimGtkConversation *
+gaim_gtk_conv_find_gtkconv(GaimConversation * conv)
+{
+	GaimBuddy *bud = gaim_find_buddy(conv->account, conv->name), *b;
+	GaimContact *c;
+	GaimBlistNode *cn;
+
+	if (!bud)
+		return NULL;
+
+	if (!(c = gaim_buddy_get_contact(bud)))
+		return NULL;
+
+	cn = (GaimBlistNode *)c;
+	for (b = (GaimBuddy *)cn->child; b; b = (GaimBuddy *) ((GaimBlistNode *)b)->next) {
+		GaimConversation *conv;
+		if ((conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, b->name, b->account))) {
+			if (conv->ui_data)
+				return conv->ui_data;
+		}
+	}
+
+	return NULL;
+}
+
+static void
+buddy_update_cb(GaimBlistNode *bnode, gpointer null)
+{
+	GList *list;
+
+	g_return_if_fail(bnode);
+	g_return_if_fail(GAIM_BLIST_NODE_IS_BUDDY(bnode));
+
+	for (list = gaim_gtk_conv_windows_get_list(); list; list = list->next)
+	{
+		GaimGtkWindow *win = list->data;
+		GaimConversation *conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+		if (gaim_conversation_get_type(conv) != GAIM_CONV_TYPE_IM)
+			continue;
+
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_MENU);
+	}
+}
+
+/**************************************************************************
+ * Conversation UI operations
+ **************************************************************************/
+static void
+private_gtkconv_new(GaimConversation *conv, gboolean hidden)
+{
+	GaimGtkConversation *gtkconv;
+	GaimConversationType conv_type = gaim_conversation_get_type(conv);
+	GtkWidget *pane = NULL;
+	GtkWidget *tab_cont;
+
+	if (conv_type == GAIM_CONV_TYPE_IM && (gtkconv = gaim_gtk_conv_find_gtkconv(conv))) {
+		conv->ui_data = gtkconv;
+		if (!g_list_find(gtkconv->convs, conv))
+			gtkconv->convs = g_list_prepend(gtkconv->convs, conv);
+		gaim_gtkconv_switch_active_conversation(conv);
+		return;
+	}
+
+	gtkconv = g_new0(GaimGtkConversation, 1);
+	conv->ui_data = gtkconv;
+	gtkconv->active_conv = conv;
+	gtkconv->convs = g_list_prepend(gtkconv->convs, conv);
+	gtkconv->send_history = g_list_append(NULL, NULL);
+
+	/* Setup some initial variables. */
+	gtkconv->sg       = gtk_size_group_new(GTK_SIZE_GROUP_BOTH);
+	gtkconv->tooltips = gtk_tooltips_new();
+	gtkconv->unseen_state = GAIM_UNSEEN_NONE;
+	gtkconv->unseen_count = 0;
+
+	if (conv_type == GAIM_CONV_TYPE_IM) {
+		gtkconv->u.im = g_malloc0(sizeof(GaimGtkImPane));
+
+		pane = setup_im_pane(gtkconv);
+	} else if (conv_type == GAIM_CONV_TYPE_CHAT) {
+		gtkconv->u.chat = g_malloc0(sizeof(GaimGtkChatPane));
+		pane = setup_chat_pane(gtkconv);
+	}
+
+	gtk_imhtml_set_format_functions(GTK_IMHTML(gtkconv->imhtml),
+			gtk_imhtml_get_format_functions(GTK_IMHTML(gtkconv->imhtml)) | GTK_IMHTML_IMAGE);
+
+	if (pane == NULL) {
+		if (conv_type == GAIM_CONV_TYPE_CHAT)
+			g_free(gtkconv->u.chat);
+		else if (conv_type == GAIM_CONV_TYPE_IM)
+			g_free(gtkconv->u.im);
+
+		g_free(gtkconv);
+		conv->ui_data = NULL;
+		return;
+	}
+
+	/* Setup drag-and-drop */
+	gtk_drag_dest_set(pane,
+	                  GTK_DEST_DEFAULT_MOTION |
+	                  GTK_DEST_DEFAULT_DROP,
+	                  te, sizeof(te) / sizeof(GtkTargetEntry),
+	                  GDK_ACTION_COPY);
+	gtk_drag_dest_set(pane,
+	                  GTK_DEST_DEFAULT_MOTION |
+	                  GTK_DEST_DEFAULT_DROP,
+	                  te, sizeof(te) / sizeof(GtkTargetEntry),
+	                  GDK_ACTION_COPY);
+	gtk_drag_dest_set(gtkconv->imhtml, 0,
+	                  te, sizeof(te) / sizeof(GtkTargetEntry),
+	                  GDK_ACTION_COPY);
+
+	gtk_drag_dest_set(gtkconv->entry, 0,
+	                  te, sizeof(te) / sizeof(GtkTargetEntry),
+	                  GDK_ACTION_COPY);
+
+	g_signal_connect(G_OBJECT(pane), "drag_data_received",
+	                 G_CALLBACK(conv_dnd_recv), gtkconv);
+	g_signal_connect(G_OBJECT(gtkconv->imhtml), "drag_data_received",
+	                 G_CALLBACK(conv_dnd_recv), gtkconv);
+	g_signal_connect(G_OBJECT(gtkconv->entry), "drag_data_received",
+	                 G_CALLBACK(conv_dnd_recv), gtkconv);
+
+	/* Setup the container for the tab. */
+	gtkconv->tab_cont = tab_cont = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	g_object_set_data(G_OBJECT(tab_cont), "GaimGtkConversation", gtkconv);
+	gtk_container_set_border_width(GTK_CONTAINER(tab_cont), GAIM_HIG_BOX_SPACE);
+	gtk_container_add(GTK_CONTAINER(tab_cont), pane);
+	gtk_widget_show(pane);
+
+	gtkconv->make_sound = TRUE;
+
+	if (gaim_prefs_get_bool("/gaim/gtk/conversations/show_formatting_toolbar"))
+		gtk_widget_show(gtkconv->toolbar);
+	else
+		gtk_widget_hide(gtkconv->toolbar);
+
+	gtk_imhtml_show_comments(GTK_IMHTML(gtkconv->imhtml),
+		gaim_prefs_get_bool("/gaim/gtk/conversations/show_timestamps"));
+	gtk_imhtml_set_protocol_name(GTK_IMHTML(gtkconv->imhtml),
+								 gaim_account_get_protocol_name(conv->account));
+
+	g_signal_connect_swapped(G_OBJECT(pane), "focus",
+	                         G_CALLBACK(gtk_widget_grab_focus),
+	                         gtkconv->entry);
+
+	if (hidden)
+		gaim_gtk_conv_window_add_gtkconv(hidden_convwin, gtkconv);
+	else
+		gaim_gtkconv_placement_place(gtkconv);
+
+	if (nick_colors == NULL) {
+		nbr_nick_colors = NUM_NICK_COLORS;
+		nick_colors = generate_nick_colors(&nbr_nick_colors, gtk_widget_get_style(gtkconv->imhtml)->base[GTK_STATE_NORMAL]);
+	}
+}
+
+static void
+gaim_gtkconv_new_hidden(GaimConversation *conv)
+{
+	private_gtkconv_new(conv, TRUE);
+}
+
+void
+gaim_gtkconv_new(GaimConversation *conv)
+{
+	private_gtkconv_new(conv, FALSE);
+}
+
+static void
+received_im_msg_cb(GaimAccount *account, char *sender, char *message,
+				   GaimConversation *conv, GaimMessageFlags flags)
+{
+	GaimConversationUiOps *ui_ops = gaim_gtk_conversations_get_conv_ui_ops();
+	if (conv != NULL)
+		return;
+
+	/* create hidden conv if hide_new pref is always */
+	if (strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/im/hide_new"), "always") == 0)
+	{
+		ui_ops->create_conversation = gaim_gtkconv_new_hidden;
+		gaim_conversation_new(GAIM_CONV_TYPE_IM, account, sender);
+		ui_ops->create_conversation = gaim_gtkconv_new;
+		return;
+	}
+
+	/* create hidden conv if hide_new pref is away and account is away */
+	if (strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/im/hide_new"), "away") == 0 &&
+	    !gaim_status_is_available(gaim_account_get_active_status(account)))
+	{
+		ui_ops->create_conversation = gaim_gtkconv_new_hidden;
+		gaim_conversation_new(GAIM_CONV_TYPE_IM, account, sender);
+		ui_ops->create_conversation = gaim_gtkconv_new;
+		return;
+	}
+}
+
+static void
+gaim_gtkconv_destroy(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtkconv->convs = g_list_remove(gtkconv->convs, conv);
+	/* Don't destroy ourselves until all our convos are gone */
+	if (gtkconv->convs)
+		return;
+
+	gaim_gtk_conv_window_remove_gtkconv(gtkconv->win, gtkconv);
+
+	/* If the "Save Conversation" or "Save Icon" dialogs are open then close them */
+	gaim_request_close_with_handle(gtkconv);
+	gaim_notify_close_with_handle(gtkconv);
+
+	gtk_widget_destroy(gtkconv->tab_cont);
+	g_object_unref(gtkconv->tab_cont);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		if (gtkconv->u.im->icon_timer != 0)
+			g_source_remove(gtkconv->u.im->icon_timer);
+
+		if (gtkconv->u.im->anim != NULL)
+			g_object_unref(G_OBJECT(gtkconv->u.im->anim));
+
+		g_free(gtkconv->u.im);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		gaim_signals_disconnect_by_handle(gtkconv->u.chat);
+		g_free(gtkconv->u.chat);
+	}
+
+	gtk_object_sink(GTK_OBJECT(gtkconv->tooltips));
+
+	gtkconv->send_history = g_list_first(gtkconv->send_history);
+	g_list_foreach(gtkconv->send_history, (GFunc)g_free, NULL);
+	g_list_free(gtkconv->send_history);
+
+	g_free(gtkconv);
+}
+
+
+static void
+gaim_gtkconv_write_im(GaimConversation *conv, const char *who,
+					  const char *message, GaimMessageFlags flags,
+					  time_t mtime)
+{
+	GaimGtkConversation *gtkconv;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	if (conv != gtkconv->active_conv &&
+	    flags & GAIM_MESSAGE_ACTIVE_ONLY)
+	{
+		/* Plugins that want these messages suppressed should be
+		 * calling gaim_conv_im_write(), so they get suppressed here,
+		 * before being written to the log. */
+		gaim_debug_info("gtkconv",
+		                "Suppressing message for an inactive conversation in gaim_gtkconv_write_im()\n");
+		return;
+	}
+
+	gaim_conversation_write(conv, who, message, flags, mtime);
+}
+
+/* The callback for an event on a link tag. */
+static gboolean buddytag_event(GtkTextTag *tag, GObject *imhtml,
+		GdkEvent *event, GtkTextIter *arg2, gpointer data) {
+	if (event->type == GDK_BUTTON_PRESS
+			|| event->type == GDK_2BUTTON_PRESS) {
+		GdkEventButton *btn_event = (GdkEventButton*) event;
+		GaimConversation *conv = data;
+		char *buddyname;
+
+		/* strlen("BUDDY ") == 6 */
+		g_return_val_if_fail((tag->name != NULL)
+				&& (strlen(tag->name) > 6), FALSE);
+
+		buddyname = (tag->name) + 6;
+
+		if (btn_event->button == 2
+				&& event->type == GDK_2BUTTON_PRESS) {
+			chat_do_info(GAIM_GTK_CONVERSATION(conv), buddyname);
+
+			return TRUE;
+		} else if (btn_event->button == 3
+				&& event->type == GDK_BUTTON_PRESS) {
+			GtkTextIter start, end;
+
+			/* we shouldn't display the popup
+			 * if the user has selected something: */
+			if (!gtk_text_buffer_get_selection_bounds(
+						gtk_text_iter_get_buffer(arg2),
+						&start, &end)) {
+				GtkWidget *menu = NULL;
+				GaimConnection *gc =
+					gaim_conversation_get_gc(conv);
+
+
+				menu = create_chat_menu(conv, buddyname, gc);
+				gtk_menu_popup(GTK_MENU(menu), NULL, NULL,
+						NULL, GTK_WIDGET(imhtml),
+						btn_event->button,
+						btn_event->time);
+
+				/* Don't propagate the event any further */
+				return TRUE;
+			}
+		}
+	}
+
+	return FALSE;
+}
+
+static GtkTextTag *get_buddy_tag(GaimConversation *conv, const char *who) {
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+	GtkTextTag *buddytag;
+	gchar *str;
+
+	str = g_strdup_printf("BUDDY %s", who);
+
+	buddytag = gtk_text_tag_table_lookup(
+			gtk_text_buffer_get_tag_table(
+				GTK_IMHTML(gtkconv->imhtml)->text_buffer), str);
+
+	if (buddytag == NULL) {
+		buddytag = gtk_text_buffer_create_tag(
+				GTK_IMHTML(gtkconv->imhtml)->text_buffer, str, NULL);
+
+		g_signal_connect(G_OBJECT(buddytag), "event",
+				G_CALLBACK(buddytag_event), conv);
+	}
+
+	g_free(str);
+
+	return buddytag;
+}
+
+static void
+gaim_gtkconv_write_conv(GaimConversation *conv, const char *name, const char *alias,
+						const char *message, GaimMessageFlags flags,
+						time_t mtime)
+{
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+	GaimConnection *gc;
+	GaimAccount *account;
+	GaimPluginProtocolInfo *prpl_info;
+	int gtk_font_options = 0;
+	int gtk_font_options_all = 0;
+	int max_scrollback_lines;
+	int line_count;
+	char buf2[BUF_LONG];
+	char *mdate;
+	char color[10];
+	char *str;
+	char *with_font_tag;
+	char *sml_attrib = NULL;
+	size_t length;
+	GaimConversationType type;
+	char *displaying;
+	gboolean plugin_return;
+	char *bracket;
+	int tag_count = 0;
+	
+	g_return_if_fail(conv != NULL);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	g_return_if_fail(gtkconv != NULL);
+
+	if (conv != gtkconv->active_conv)
+	{
+		if (flags & GAIM_MESSAGE_ACTIVE_ONLY)
+		{
+			/* Unless this had GAIM_MESSAGE_NO_LOG, this message
+			 * was logged.  Plugin writers: if this isn't what
+			 * you wanted, call gaim_conv_im_write() instead of
+			 * gaim_conversation_write(). */
+			gaim_debug_info("gtkconv",
+			                "Suppressing message for an inactive conversation in gaim_gtkconv_write_conv()\n");
+			return;
+		}
+
+		/* Set the active conversation to the one that just messaged us. */
+		/* TODO: consider not doing this if the account is offline or something */
+		if (flags & (GAIM_MESSAGE_SEND | GAIM_MESSAGE_RECV))
+			gaim_gtkconv_switch_active_conversation(conv);
+	}
+
+	type = gaim_conversation_get_type(conv);
+	account = gaim_conversation_get_account(conv);
+	g_return_if_fail(account != NULL);
+	gc = gaim_account_get_connection(account);
+	g_return_if_fail(gc != NULL);
+
+	displaying = g_strdup(message);
+	plugin_return = GPOINTER_TO_INT(gaim_signal_emit_return_1(
+							gaim_gtk_conversations_get_handle(), (type == GAIM_CONV_TYPE_IM ?
+							"displaying-im-msg" : "displaying-chat-msg"),
+							account, name, &displaying, conv, flags));
+	if (plugin_return)
+	{
+		g_free(displaying);
+		return;
+	}
+	message = displaying;
+	length = strlen(message) + 1;
+
+	/* Awful hack to work around GtkIMHtml's inefficient rendering of messages with lots of formatting changes.
+	 * If a message has over 100 '<' characters, strip formatting before appending it. Hopefully nobody actually
+	 * needs that much formatting, anyway.
+	 */
+	for (bracket = strchr(message, '<'); bracket && *(bracket + 1); bracket = strchr(bracket + 1, '<'))
+		tag_count++;
+	
+	if (tag_count > 100) {
+		char *tmp = message;
+		message = displaying = gaim_markup_strip_html(message);
+		g_free(tmp);
+	}	
+	
+	win = gtkconv->win;
+	prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	line_count = gtk_text_buffer_get_line_count(
+			gtk_text_view_get_buffer(GTK_TEXT_VIEW(
+				gtkconv->imhtml)));
+
+	max_scrollback_lines = gaim_prefs_get_int(
+		"/gaim/gtk/conversations/scrollback_lines");
+	/* If we're sitting at more than 100 lines more than the
+	   max scrollback, trim down to max scrollback */
+	if (max_scrollback_lines > 0
+			&& line_count > (max_scrollback_lines + 100)) {
+		GtkTextBuffer *text_buffer = gtk_text_view_get_buffer(
+			GTK_TEXT_VIEW(gtkconv->imhtml));
+		GtkTextIter start, end;
+
+		gtk_text_buffer_get_start_iter(text_buffer, &start);
+		gtk_text_buffer_get_iter_at_line(text_buffer, &end,
+			(line_count - max_scrollback_lines));
+		gtk_imhtml_delete(GTK_IMHTML(gtkconv->imhtml), &start, &end);
+	}
+
+	if (type == GAIM_CONV_TYPE_CHAT)
+	{
+		/* Create anchor for user */
+		GtkTextIter iter;
+		char *tmp = g_strconcat("user:", name, NULL);
+
+		gtk_text_buffer_get_end_iter(gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->imhtml)), &iter);
+		gtk_text_buffer_create_mark(gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->imhtml)),
+								tmp, &iter, TRUE);
+		g_free(tmp);
+	}
+
+	if (gaim_prefs_get_bool("/gaim/gtk/conversations/use_smooth_scrolling"))
+		gtk_font_options_all |= GTK_IMHTML_USE_SMOOTHSCROLLING;
+
+	if (gtk_text_buffer_get_char_count(gtk_text_view_get_buffer(GTK_TEXT_VIEW(gtkconv->imhtml))))
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), "<BR>", gtk_font_options_all);
+
+	mdate = gaim_signal_emit_return_1(gaim_gtk_conversations_get_handle(),
+	                                  "conversation-timestamp",
+	                                  conv, mtime);
+	if (mdate == NULL)
+	{
+		struct tm *tm = localtime(&mtime);
+		if (time(NULL) > mtime + 20*60) /* show date if older than 20 minutes */
+			mdate = g_strdup(gaim_date_format_long(tm));
+		else
+			mdate = g_strdup(gaim_time_format(tm));
+	}
+
+	sml_attrib = g_strdup_printf("sml=\"%s\"", gaim_account_get_protocol_name(account));
+
+	gtk_font_options |= GTK_IMHTML_NO_COMMENTS;
+
+	if ((flags & GAIM_MESSAGE_RECV) &&
+			!gaim_prefs_get_bool("/gaim/gtk/conversations/show_incoming_formatting"))
+		gtk_font_options |= GTK_IMHTML_NO_COLOURS | GTK_IMHTML_NO_FONTS | GTK_IMHTML_NO_SIZES | GTK_IMHTML_NO_FORMATTING;
+
+	/* this is gonna crash one day, I can feel it. */
+	if (GAIM_PLUGIN_PROTOCOL_INFO(gaim_find_prpl(gaim_account_get_protocol_id(conv->account)))->options &
+	    OPT_PROTO_USE_POINTSIZE) {
+		gtk_font_options |= GTK_IMHTML_USE_POINTSIZE;
+	}
+
+
+	/* TODO: These colors should not be hardcoded so log.c can use them */
+	if (flags & GAIM_MESSAGE_SYSTEM) {
+		g_snprintf(buf2, sizeof(buf2),
+			   "<FONT %s><FONT SIZE=\"2\"><!--(%s) --></FONT><B>%s</B></FONT>",
+			   sml_attrib ? sml_attrib : "", mdate, message);
+
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), buf2, gtk_font_options_all);
+
+	} else if (flags & GAIM_MESSAGE_ERROR) {
+		g_snprintf(buf2, sizeof(buf2),
+			   "<FONT COLOR=\"#ff0000\"><FONT %s><FONT SIZE=\"2\"><!--(%s) --></FONT><B>%s</B></FONT></FONT>",
+			   sml_attrib ? sml_attrib : "", mdate, message);
+
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), buf2, gtk_font_options_all);
+
+	} else if (flags & GAIM_MESSAGE_NO_LOG) {
+		g_snprintf(buf2, BUF_LONG,
+			   "<B><FONT %s COLOR=\"#777777\">%s</FONT></B>",
+			   sml_attrib ? sml_attrib : "", message);
+
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), buf2, gtk_font_options_all);
+	} else if (flags & GAIM_MESSAGE_RAW) {
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), message, gtk_font_options_all);
+	} else {
+		char *new_message = g_memdup(message, length);
+		char *alias_escaped = (alias ? g_markup_escape_text(alias, strlen(alias)) : g_strdup(""));
+		/* The initial offset is to deal with
+		 * escaped entities making the string longer */
+		int tag_start_offset = alias ? (strlen(alias_escaped) - strlen(alias)) : 0;
+		int tag_end_offset = 0;
+		GtkSmileyTree *tree = NULL;
+		GHashTable *smiley_data = NULL;
+
+		if (flags & GAIM_MESSAGE_SEND)
+		{
+			/* Temporarily revert to the original smiley-data to avoid showing up
+			 * custom smileys of the buddy when sending message
+			 */
+			tree = GTK_IMHTML(gtkconv->imhtml)->default_smilies;
+			GTK_IMHTML(gtkconv->imhtml)->default_smilies =
+									GTK_IMHTML(gtkconv->entry)->default_smilies;
+			smiley_data = GTK_IMHTML(gtkconv->imhtml)->smiley_data;
+			GTK_IMHTML(gtkconv->imhtml)->smiley_data = GTK_IMHTML(gtkconv->entry)->smiley_data;
+		}
+
+		if (flags & GAIM_MESSAGE_WHISPER) {
+			str = g_malloc(1024);
+
+			/* If we're whispering, it's not an autoresponse. */
+			if (gaim_message_meify(new_message, -1 )) {
+				g_snprintf(str, 1024, "***%s", alias_escaped);
+				strcpy(color, "#6C2585");
+				tag_start_offset += 3;
+			}
+			else {
+				g_snprintf(str, 1024, "*%s*:", alias_escaped);
+				tag_start_offset += 1;
+				tag_end_offset = 2;
+				strcpy(color, "#00FF00");
+			}
+		}
+		else {
+			if (gaim_message_meify(new_message, -1)) {
+				str = g_malloc(1024);
+
+				if (flags & GAIM_MESSAGE_AUTO_RESP) {
+					g_snprintf(str, 1024, "%s ***%s", AUTO_RESPONSE, alias_escaped);
+					tag_start_offset += 4
+						+ strlen(AUTO_RESPONSE);
+				} else {
+					g_snprintf(str, 1024, "***%s", alias_escaped);
+					tag_start_offset += 3;
+				}
+
+				if (flags & GAIM_MESSAGE_NICK)
+					strcpy(color, HIGHLIGHT_COLOR);
+				else
+					strcpy(color, "#062585");
+			}
+			else {
+				str = g_malloc(1024);
+				if (flags & GAIM_MESSAGE_AUTO_RESP) {
+					g_snprintf(str, 1024, "%s %s", alias_escaped, AUTO_RESPONSE);
+					tag_start_offset += 1
+						+ strlen(AUTO_RESPONSE);
+				} else {
+					g_snprintf(str, 1024, "%s:", alias_escaped);
+					tag_end_offset = 1;
+				}
+				if (flags & GAIM_MESSAGE_NICK)
+					strcpy(color, HIGHLIGHT_COLOR);
+				else if (flags & GAIM_MESSAGE_RECV) {
+					if (type == GAIM_CONV_TYPE_CHAT) {
+						GdkColor *col = get_nick_color(gtkconv, name);
+
+						g_snprintf(color, sizeof(color), "#%02X%02X%02X",
+							   col->red >> 8, col->green >> 8, col->blue >> 8);
+					} else
+						strcpy(color, RECV_COLOR);
+				}
+				else if (flags & GAIM_MESSAGE_SEND)
+					strcpy(color, SEND_COLOR);
+				else {
+					gaim_debug_error("gtkconv", "message missing flags\n");
+					strcpy(color, "#000000");
+				}
+			}
+		}
+
+		g_free(alias_escaped);
+
+		/* Are we in a chat where we can tell which users are buddies? */
+		if  (!(prpl_info->options & OPT_PROTO_UNIQUE_CHATNAME) &&
+		     gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+
+			/* Bold buddies to make them stand out from non-buddies. */
+			if (flags & GAIM_MESSAGE_SEND ||
+			    flags & GAIM_MESSAGE_NICK ||
+			    gaim_find_buddy(account, name) != NULL) {
+				g_snprintf(buf2, BUF_LONG,
+					   "<FONT COLOR=\"%s\" %s><FONT SIZE=\"2\"><!--(%s) --></FONT>"
+					   "<B>%s</B></FONT> ",
+					   color, sml_attrib ? sml_attrib : "", mdate, str);
+			} else {
+				g_snprintf(buf2, BUF_LONG,
+					   "<FONT COLOR=\"%s\" %s><FONT SIZE=\"2\"><!--(%s) --></FONT>"
+					   "%s</FONT> ",
+					   color, sml_attrib ? sml_attrib : "", mdate, str);
+
+			}
+		} else {
+			/* Bold everyone's name to make the name stand out from the message. */
+			g_snprintf(buf2, BUF_LONG,
+				   "<FONT COLOR=\"%s\" %s><FONT SIZE=\"2\"><!--(%s) --></FONT>"
+				   "<B>%s</B></FONT> ",
+				   color, sml_attrib ? sml_attrib : "", mdate, str);
+		}
+
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml), buf2, gtk_font_options_all);
+
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT &&
+		    !(flags & GAIM_MESSAGE_SEND)) {
+
+			GtkTextIter start, end;
+			GtkTextTag *buddytag = get_buddy_tag(conv, name);
+
+			gtk_text_buffer_get_end_iter(
+					GTK_IMHTML(gtkconv->imhtml)->text_buffer,
+					&end);
+			gtk_text_iter_backward_chars(&end,
+					tag_end_offset + 1);
+
+			gtk_text_buffer_get_end_iter(
+					GTK_IMHTML(gtkconv->imhtml)->text_buffer,
+					&start);
+			gtk_text_iter_backward_chars(&start,
+					strlen(str) + 1 - tag_start_offset);
+
+			gtk_text_buffer_apply_tag(
+					GTK_IMHTML(gtkconv->imhtml)->text_buffer,
+					buddytag, &start, &end);
+		}
+
+		g_free(str);
+
+		if(gc){
+			char *pre = g_strdup_printf("<font %s>", sml_attrib ? sml_attrib : "");
+			char *post = "</font>";
+			int pre_len = strlen(pre);
+			int post_len = strlen(post);
+
+			with_font_tag = g_malloc(length + pre_len + post_len + 1);
+
+			strcpy(with_font_tag, pre);
+			memcpy(with_font_tag + pre_len, new_message, length);
+			strcpy(with_font_tag + pre_len + length, post);
+
+			length += pre_len + post_len;
+			g_free(pre);
+		}
+		else
+			with_font_tag = g_memdup(new_message, length);
+
+		gtk_imhtml_append_text(GTK_IMHTML(gtkconv->imhtml),
+							 with_font_tag, gtk_font_options | gtk_font_options_all);
+
+		if (flags & GAIM_MESSAGE_SEND)
+		{
+			/* Restore the smiley-data */
+			GTK_IMHTML(gtkconv->imhtml)->default_smilies = tree;
+			GTK_IMHTML(gtkconv->imhtml)->smiley_data = smiley_data;
+		}
+
+		g_free(with_font_tag);
+		g_free(new_message);
+	}
+
+	g_free(mdate);
+	g_free(sml_attrib);
+
+	/* Tab highlighting stuff */
+	if (!(flags & GAIM_MESSAGE_SEND) && !gaim_gtkconv_has_focus(conv))
+	{
+		GaimUnseenState unseen = GAIM_UNSEEN_NONE;
+
+		if ((flags & GAIM_MESSAGE_NICK) == GAIM_MESSAGE_NICK)
+			unseen = GAIM_UNSEEN_NICK;
+		else if (((flags & GAIM_MESSAGE_SYSTEM) == GAIM_MESSAGE_SYSTEM) ||
+			  ((flags & GAIM_MESSAGE_ERROR) == GAIM_MESSAGE_ERROR))
+			unseen = GAIM_UNSEEN_EVENT;
+		else if ((flags & GAIM_MESSAGE_NO_LOG) == GAIM_MESSAGE_NO_LOG)
+			unseen = GAIM_UNSEEN_NO_LOG;
+		else
+			unseen = GAIM_UNSEEN_TEXT;
+
+		gtkconv_set_unseen(gtkconv, unseen);
+	}
+
+	gaim_signal_emit(gaim_gtk_conversations_get_handle(),
+		(type == GAIM_CONV_TYPE_IM ? "displayed-im-msg" : "displayed-chat-msg"),
+		account, name, message, conv, flags);
+	g_free(displaying);
+}
+static void
+gaim_gtkconv_chat_add_users(GaimConversation *conv, GList *cbuddies, gboolean new_arrivals)
+{
+	GaimConvChat *chat;
+	GaimGtkConversation *gtkconv;
+	GaimGtkChatPane *gtkchat;
+	GtkListStore *ls;
+	GList *l;
+
+	char tmp[BUF_LONG];
+	int num_users;
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+
+	num_users = g_list_length(gaim_conv_chat_get_users(chat));
+
+	g_snprintf(tmp, sizeof(tmp),
+			   ngettext("%d person in room", "%d people in room",
+						num_users),
+			   num_users);
+
+	gtk_label_set_text(GTK_LABEL(gtkchat->count), tmp);
+
+	ls = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list)));
+
+#if GTK_CHECK_VERSION(2,6,0)
+	gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(ls),  GTK_TREE_SORTABLE_UNSORTED_SORT_COLUMN_ID,
+										 GTK_TREE_SORTABLE_UNSORTED_SORT_COLUMN_ID);
+#endif
+
+	l = cbuddies;
+	while (l != NULL) {
+		add_chat_buddy_common(conv, (GaimConvChatBuddy *)l->data, NULL);
+		l = l->next;
+	}
+
+	/* Currently GTK+ maintains our sorted list after it's in the tree.
+	 * This may change if it turns out we can manage it faster ourselves.
+ 	 */
+	gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(ls),  CHAT_USERS_ALIAS_KEY_COLUMN,
+										 GTK_SORT_ASCENDING);
+}
+
+static void
+gaim_gtkconv_chat_rename_user(GaimConversation *conv, const char *old_name,
+			      const char *new_name, const char *new_alias)
+{
+	GaimConvChat *chat;
+	GaimGtkConversation *gtkconv;
+	GaimGtkChatPane *gtkchat;
+	GaimConvChatBuddyFlags flags;
+	GaimConvChatBuddy *cbuddy;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	int f = 1;
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+
+	if (!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+		return;
+
+	while (f != 0) {
+		char *val;
+
+		gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_NAME_COLUMN, &val, CHAT_USERS_FLAGS_COLUMN, &flags, -1);
+
+		if (!gaim_utf8_strcasecmp(old_name, val)) {
+			gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
+			g_free(val);
+			break;
+		}
+
+		f = gtk_tree_model_iter_next(GTK_TREE_MODEL(model), &iter);
+
+		g_free(val);
+	}
+
+	if (!gaim_conv_chat_find_user(chat, old_name))
+		return;
+
+	g_return_if_fail(new_alias != NULL);
+
+	cbuddy = gaim_conv_chat_cb_new(new_name, new_alias, flags);
+
+	add_chat_buddy_common(conv, cbuddy, old_name);
+}
+
+static void
+gaim_gtkconv_chat_remove_users(GaimConversation *conv, GList *users)
+{
+	GaimConvChat *chat;
+	GaimGtkConversation *gtkconv;
+	GaimGtkChatPane *gtkchat;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	GList *l;
+	char tmp[BUF_LONG];
+	int num_users;
+	gboolean f;
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+
+	num_users = g_list_length(gaim_conv_chat_get_users(chat));
+
+	for (l = users; l != NULL; l = l->next) {
+		model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+
+		if (!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+			continue;
+
+		do {
+			char *val;
+
+			gtk_tree_model_get(GTK_TREE_MODEL(model), &iter,
+							   CHAT_USERS_NAME_COLUMN, &val, -1);
+
+			if (!gaim_utf8_strcasecmp((char *)l->data, val)) {
+#if GTK_CHECK_VERSION(2,2,0)
+				f = gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
+#else
+				gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
+				f = gtk_tree_model_iter_next(GTK_TREE_MODEL(model), &iter);
+#endif
+			}
+			else
+				f = gtk_tree_model_iter_next(GTK_TREE_MODEL(model), &iter);
+
+			g_free(val);
+		} while (f);
+	}
+
+	g_snprintf(tmp, sizeof(tmp),
+			   ngettext("%d person in room", "%d people in room",
+						num_users), num_users);
+
+	gtk_label_set_text(GTK_LABEL(gtkchat->count), tmp);
+}
+
+static void
+gaim_gtkconv_chat_update_user(GaimConversation *conv, const char *user)
+{
+	GaimConvChat *chat;
+	GaimConvChatBuddyFlags flags;
+	GaimConvChatBuddy *cbuddy;
+	GaimGtkConversation *gtkconv;
+	GaimGtkChatPane *gtkchat;
+	GtkTreeIter iter;
+	GtkTreeModel *model;
+	int f = 1;
+	char *alias = NULL;
+
+	chat    = GAIM_CONV_CHAT(conv);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	gtkchat = gtkconv->u.chat;
+
+	model = gtk_tree_view_get_model(GTK_TREE_VIEW(gtkchat->list));
+
+	if (!gtk_tree_model_get_iter_first(GTK_TREE_MODEL(model), &iter))
+		return;
+
+	while (f != 0) {
+		char *val;
+
+		gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_NAME_COLUMN, &val, -1);
+
+		if (!gaim_utf8_strcasecmp(user, val)) {
+			gtk_tree_model_get(GTK_TREE_MODEL(model), &iter, CHAT_USERS_ALIAS_COLUMN, &alias, -1);
+			gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
+			g_free(val);
+			break;
+		}
+
+		f = gtk_tree_model_iter_next(GTK_TREE_MODEL(model), &iter);
+
+		g_free(val);
+	}
+
+	if (!gaim_conv_chat_find_user(chat, user))
+	{
+		g_free(alias);
+		return;
+	}
+
+	g_return_if_fail(alias != NULL);
+
+	flags = gaim_conv_chat_user_get_flags(chat, user);
+
+	cbuddy = gaim_conv_chat_cb_new(user, alias, flags);
+
+	add_chat_buddy_common(conv, cbuddy, NULL);
+	g_free(alias);
+}
+
+gboolean
+gaim_gtkconv_has_focus(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+	GaimGtkWindow *win;
+	gboolean has_focus;
+
+	win = gtkconv->win;
+
+	g_object_get(G_OBJECT(win->window), "has-toplevel-focus", &has_focus, NULL);
+
+	if (has_focus && gaim_gtk_conv_window_is_active_conversation(conv))
+		return TRUE;
+
+	return FALSE;
+}
+
+static void gaim_gtkconv_custom_smiley_allocated(GdkPixbufLoader *loader, gpointer user_data)
+{
+	GtkIMHtmlSmiley *smiley;
+
+	smiley = (GtkIMHtmlSmiley *)user_data;
+	smiley->icon = gdk_pixbuf_loader_get_animation(loader);
+
+	if (smiley->icon)
+		g_object_ref(G_OBJECT(smiley->icon));
+#ifdef DEBUG_CUSTOM_SMILEY
+	gaim_debug_info("custom-smiley", "gaim_gtkconv_custom_smiley_allocated(): got GdkPixbufAnimation %p for smiley '%s'\n", smiley->icon, smiley->smile);
+#endif
+}
+
+static void gaim_gtkconv_custom_smiley_closed(GdkPixbufLoader *loader, gpointer user_data)
+{
+	GtkIMHtmlSmiley *smiley;
+	GtkWidget *icon = NULL;
+	GtkTextChildAnchor *anchor = NULL;
+	GSList *current = NULL;
+
+	smiley = (GtkIMHtmlSmiley *)user_data;
+	if (!smiley->imhtml) {
+#ifdef DEBUG_CUSTOM_SMILEY
+		gaim_debug_error("custom-smiley", "gaim_gtkconv_custom_smiley_closed(): orphan smiley found: %p\n", smiley);
+#endif
+		g_object_unref(G_OBJECT(loader));
+		smiley->loader = NULL;
+		return;
+	}
+
+	for (current = smiley->anchors; current; current = g_slist_next(current)) {
+
+		icon = gtk_image_new_from_animation(smiley->icon);
+
+#ifdef DEBUG_CUSTOM_SMILEY
+		gaim_debug_info("custom-smiley", "gaim_gtkconv_custom_smiley_closed(): got GtkImage %p from GtkPixbufAnimation %p for smiley '%s'\n",
+				icon, smiley->icon, smiley->smile);
+#endif
+		if (icon) {
+			gtk_widget_show(icon);
+
+			anchor = GTK_TEXT_CHILD_ANCHOR(current->data);
+
+			g_object_set_data_full(G_OBJECT(anchor), "gtkimhtml_plaintext", gaim_unescape_html(smiley->smile), g_free);
+			g_object_set_data_full(G_OBJECT(anchor), "gtkimhtml_htmltext", g_strdup(smiley->smile), g_free);
+
+			if (smiley->imhtml)
+				gtk_text_view_add_child_at_anchor(GTK_TEXT_VIEW(smiley->imhtml), icon, anchor);
+		}
+
+	}
+
+	g_slist_free(smiley->anchors);
+	smiley->anchors = NULL;
+
+	g_object_unref(G_OBJECT(loader));
+	smiley->loader = NULL;
+}
+
+static gboolean
+add_custom_smiley_for_imhtml(GtkIMHtml *imhtml, const char *sml, const char *smile)
+{
+	GtkIMHtmlSmiley *smiley;
+	GdkPixbufLoader *loader;
+
+	smiley = gtk_imhtml_smiley_get(imhtml, sml, smile);
+
+	if (smiley) {
+
+		if (!(smiley->flags & GTK_IMHTML_SMILEY_CUSTOM)) {
+			return FALSE;
+		}
+
+		/* Close the old GdkPixbufAnimation, then create a new one for
+		 * the smiley we are about to receive */
+		g_object_unref(G_OBJECT(smiley->icon));
+
+		/* XXX: Is it necessary to _unref the loader first? */
+		smiley->loader = gdk_pixbuf_loader_new();
+		smiley->icon = NULL;
+
+		g_signal_connect(smiley->loader, "area_prepared", G_CALLBACK(gaim_gtkconv_custom_smiley_allocated), smiley);
+		g_signal_connect(smiley->loader, "closed", G_CALLBACK(gaim_gtkconv_custom_smiley_closed), smiley);
+
+		return TRUE;
+	}
+
+	loader = gdk_pixbuf_loader_new();
+
+	/* this is wrong, this file ought not call g_new on GtkIMHtmlSmiley */
+	/* Let gtk_imhtml have a gtk_imhtml_smiley_new function, and let
+	   GtkIMHtmlSmiley by opaque */
+	smiley = g_new0(GtkIMHtmlSmiley, 1);
+	smiley->file   = NULL;
+	smiley->smile  = g_strdup(smile);
+	smiley->loader = loader;
+	smiley->flags  = smiley->flags | GTK_IMHTML_SMILEY_CUSTOM;
+
+	g_signal_connect(smiley->loader, "area_prepared", G_CALLBACK(gaim_gtkconv_custom_smiley_allocated), smiley);
+	g_signal_connect(smiley->loader, "closed", G_CALLBACK(gaim_gtkconv_custom_smiley_closed), smiley);
+
+	gtk_imhtml_associate_smiley(imhtml, sml, smiley);
+
+	return TRUE;
+}
+
+static gboolean
+gaim_gtkconv_custom_smiley_add(GaimConversation *conv, const char *smile, gboolean remote)
+{
+	GaimGtkConversation *gtkconv;
+	struct smiley_list *list;
+	const char *sml = NULL, *conv_sml;
+
+	if (!conv || !smile || !*smile) {
+		return FALSE;
+	}
+
+	/* If smileys are off, return false */
+	if (gaim_gtkthemes_smileys_disabled())
+		return FALSE;
+
+	/* If possible add this smiley to the current theme.
+	 * The addition is only temporary: custom smilies aren't saved to disk. */
+	conv_sml = gaim_account_get_protocol_name(conv->account);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	for (list = (struct smiley_list *)current_smiley_theme->list; list; list = list->next) {
+		if (!strcmp(list->sml, conv_sml)) {
+			sml = list->sml;
+			break;
+		}
+	}
+
+	if (!add_custom_smiley_for_imhtml(GTK_IMHTML(gtkconv->imhtml), sml, smile))
+		return FALSE;
+
+	if (!remote)	/* If it's a local custom smiley, then add it for the entry */
+		if (!add_custom_smiley_for_imhtml(GTK_IMHTML(gtkconv->entry), sml, smile))
+			return FALSE;
+
+	return TRUE;
+}
+
+static void
+gaim_gtkconv_custom_smiley_write(GaimConversation *conv, const char *smile,
+                                      const guchar *data, gsize size)
+{
+	GaimGtkConversation *gtkconv;
+	GtkIMHtmlSmiley *smiley;
+	GdkPixbufLoader *loader;
+	const char *sml;
+
+	sml = gaim_account_get_protocol_name(conv->account);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	smiley = gtk_imhtml_smiley_get(GTK_IMHTML(gtkconv->imhtml), sml, smile);
+
+	if (!smiley)
+		return;
+
+	loader = smiley->loader;
+	if (!loader)
+		return;
+
+	gdk_pixbuf_loader_write(loader, data, size, NULL);
+}
+
+static void
+gaim_gtkconv_custom_smiley_close(GaimConversation *conv, const char *smile)
+{
+	GaimGtkConversation *gtkconv;
+	GtkIMHtmlSmiley *smiley;
+	GdkPixbufLoader *loader;
+	const char *sml;
+
+	g_return_if_fail(conv  != NULL);
+	g_return_if_fail(smile != NULL);
+
+	sml = gaim_account_get_protocol_name(conv->account);
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	smiley = gtk_imhtml_smiley_get(GTK_IMHTML(gtkconv->imhtml), sml, smile);
+
+	if (!smiley)
+		return;
+
+	loader = smiley->loader;
+
+	if (!loader)
+		return;
+
+
+
+	gaim_debug_info("gtkconv", "About to close the smiley pixbuf\n");
+
+	gdk_pixbuf_loader_close(loader, NULL);
+
+}
+
+static void
+gaim_gtkconv_send_confirm(GaimConversation *conv, const char *message)
+{
+	GaimGtkConversation *gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	gtk_imhtml_append_text(GTK_IMHTML(gtkconv->entry), message, 0);
+}
+
+/*
+ * Makes sure all the menu items and all the buttons are hidden/shown and
+ * sensitive/insensitive.  This is called after changing tabs and when an
+ * account signs on or off.
+ */
+static void
+gray_stuff_out(GaimGtkConversation *gtkconv)
+{
+	GaimGtkWindow *win;
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimConnection *gc;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+	GdkPixbuf *window_icon = NULL;
+	GtkIMHtmlButtons buttons;
+	GaimAccount *account;
+
+	win     = gaim_gtkconv_get_window(gtkconv);
+	gc      = gaim_conversation_get_gc(conv);
+	account = gaim_conversation_get_account(conv);
+
+	if (gc != NULL)
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl);
+
+	if (win->menu.send_to != NULL)
+		update_send_to_selection(win);
+
+	/*
+	 * Handle hiding and showing stuff based on what type of conv this is.
+	 * Stuff that Gaim IMs support in general should be shown for IM
+	 * conversations.  Stuff that Gaim chats support in general should be
+	 * shown for chat conversations.  It doesn't matter whether the PRPL
+	 * supports it or not--that only affects if the button or menu item
+	 * is sensitive or not.
+	 */
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) {
+		/* Show stuff that applies to IMs, hide stuff that applies to chats */
+
+		/* Deal with menu items */
+		gtk_widget_show(win->menu.view_log);
+		gtk_widget_show(win->menu.send_file);
+		gtk_widget_show(win->menu.add_pounce);
+		gtk_widget_show(win->menu.get_info);
+		gtk_widget_hide(win->menu.invite);
+		gtk_widget_show(win->menu.alias);
+		gtk_widget_show(win->menu.block);
+
+		if ((account == NULL) || gaim_find_buddy(account, gaim_conversation_get_name(conv)) == NULL) {
+			gtk_widget_show(win->menu.add);
+			gtk_widget_hide(win->menu.remove);
+		} else {
+			gtk_widget_show(win->menu.remove);
+			gtk_widget_hide(win->menu.add);
+		}
+
+		gtk_widget_show(win->menu.insert_link);
+		gtk_widget_show(win->menu.insert_image);
+		gtk_widget_show(win->menu.show_icon);
+	} else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT) {
+		/* Show stuff that applies to Chats, hide stuff that applies to IMs */
+
+		/* Deal with menu items */
+		gtk_widget_show(win->menu.view_log);
+		gtk_widget_hide(win->menu.send_file);
+		gtk_widget_hide(win->menu.add_pounce);
+		gtk_widget_hide(win->menu.get_info);
+		gtk_widget_show(win->menu.invite);
+		gtk_widget_show(win->menu.alias);
+		gtk_widget_hide(win->menu.block);
+		gtk_widget_hide(win->menu.show_icon);
+
+		if ((account == NULL) || gaim_blist_find_chat(account, gaim_conversation_get_name(conv)) == NULL) {
+			/* If the chat is NOT in the buddy list */
+			gtk_widget_show(win->menu.add);
+			gtk_widget_hide(win->menu.remove);
+		} else {
+			/* If the chat IS in the buddy list */
+			gtk_widget_hide(win->menu.add);
+			gtk_widget_show(win->menu.remove);
+		}
+
+		gtk_widget_show(win->menu.insert_link);
+		gtk_widget_hide(win->menu.insert_image);
+	}
+
+	/*
+	 * Handle graying stuff out based on whether an account is connected
+	 * and what features that account supports.
+	 */
+	if ((gc != NULL) &&
+		((gaim_conversation_get_type(conv) != GAIM_CONV_TYPE_CHAT) ||
+		 !gaim_conv_chat_has_left(GAIM_CONV_CHAT(conv)) ))
+	{
+		/* Account is online */
+		/* Deal with the toolbar */
+		if (conv->features & GAIM_CONNECTION_HTML)
+		{
+			buttons = GTK_IMHTML_ALL; /* Everything on */
+			if (conv->features & GAIM_CONNECTION_NO_BGCOLOR)
+				buttons &= ~GTK_IMHTML_BACKCOLOR;
+			if (conv->features & GAIM_CONNECTION_NO_FONTSIZE)
+			{
+				buttons &= ~GTK_IMHTML_GROW;
+				buttons &= ~GTK_IMHTML_SHRINK;
+			}
+			if (conv->features & GAIM_CONNECTION_NO_URLDESC)
+				buttons &= ~GTK_IMHTML_LINKDESC;
+		} else {
+			buttons = GTK_IMHTML_SMILEY | GTK_IMHTML_IMAGE;
+		}
+
+		if (!(prpl_info->options & OPT_PROTO_IM_IMAGE) ||
+				conv->features & GAIM_CONNECTION_NO_IMAGES)
+			buttons &= ~GTK_IMHTML_IMAGE;
+
+		gtk_imhtml_set_format_functions(GTK_IMHTML(gtkconv->entry), buttons);
+		if (account != NULL)
+			gtk_imhtmltoolbar_associate_smileys(GTK_IMHTMLTOOLBAR(gtkconv->toolbar), gaim_account_get_protocol_id(account));
+
+		/* Deal with menu items */
+		gtk_widget_set_sensitive(win->menu.view_log, TRUE);
+		gtk_widget_set_sensitive(win->menu.add_pounce, TRUE);
+		gtk_widget_set_sensitive(win->menu.get_info, (prpl_info->get_info != NULL));
+		gtk_widget_set_sensitive(win->menu.invite, (prpl_info->chat_invite != NULL));
+		gtk_widget_set_sensitive(win->menu.insert_link, (conv->features & GAIM_CONNECTION_HTML));
+		gtk_widget_set_sensitive(win->menu.insert_image, (prpl_info->options & OPT_PROTO_IM_IMAGE) && !(conv->features & GAIM_CONNECTION_NO_IMAGES));
+
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		{
+			gtk_widget_set_sensitive(win->menu.add, (prpl_info->add_buddy != NULL));
+			gtk_widget_set_sensitive(win->menu.remove, (prpl_info->remove_buddy != NULL));
+			gtk_widget_set_sensitive(win->menu.send_file,
+									 (prpl_info->send_file != NULL && (!prpl_info->can_receive_file ||
+									  prpl_info->can_receive_file(gc, gaim_conversation_get_name(conv)))));
+			gtk_widget_set_sensitive(win->menu.alias,
+									 (account != NULL) &&
+									 (gaim_find_buddy(account, gaim_conversation_get_name(conv)) != NULL));
+		}
+		else if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+		{
+			gtk_widget_set_sensitive(win->menu.add, (prpl_info->join_chat != NULL));
+			gtk_widget_set_sensitive(win->menu.remove, (prpl_info->join_chat != NULL));
+			gtk_widget_set_sensitive(win->menu.alias,
+									 (account != NULL) &&
+									 (gaim_blist_find_chat(account, gaim_conversation_get_name(conv)) != NULL));
+		}
+
+	} else {
+		/* Account is offline */
+		/* Or it's a chat that we've left. */
+
+		/* Then deal with menu items */
+		gtk_widget_set_sensitive(win->menu.view_log, TRUE);
+		gtk_widget_set_sensitive(win->menu.send_file, FALSE);
+		gtk_widget_set_sensitive(win->menu.add_pounce, TRUE);
+		gtk_widget_set_sensitive(win->menu.get_info, FALSE);
+		gtk_widget_set_sensitive(win->menu.invite, FALSE);
+		gtk_widget_set_sensitive(win->menu.alias, FALSE);
+		gtk_widget_set_sensitive(win->menu.add, FALSE);
+		gtk_widget_set_sensitive(win->menu.remove, FALSE);
+		gtk_widget_set_sensitive(win->menu.insert_link, TRUE);
+		gtk_widget_set_sensitive(win->menu.insert_image, FALSE);
+	}
+
+	/*
+	 * Update the window's icon
+	 */
+	if (gaim_gtk_conv_window_is_active_conversation(conv))
+	{
+		if ((gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) &&
+				(gtkconv->u.im->anim))
+		{
+			window_icon =
+				gdk_pixbuf_animation_get_static_image(gtkconv->u.im->anim);
+			g_object_ref(window_icon);
+		} else {
+			window_icon = gaim_gtkconv_get_tab_icon(conv, FALSE);
+		}
+		gtk_window_set_icon(GTK_WINDOW(win->window), window_icon);
+		if (window_icon != NULL)
+			g_object_unref(G_OBJECT(window_icon));
+	}
+}
+
+static void
+gaim_gtkconv_update_fields(GaimConversation *conv, GaimGtkConvFields fields)
+{
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	if (!gtkconv)
+		return;
+	win = gaim_gtkconv_get_window(gtkconv);
+	if (!win)
+		return;
+
+	if (fields & GAIM_GTKCONV_SET_TITLE)
+	{
+		gaim_conversation_autoset_title(conv);
+	}
+
+	if (fields & GAIM_GTKCONV_BUDDY_ICON)
+	{
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+			gaim_gtkconv_update_buddy_icon(conv);
+	}
+
+	if (fields & GAIM_GTKCONV_MENU)
+	{
+		gray_stuff_out(GAIM_GTK_CONVERSATION(conv));
+		generate_send_to_items(win);
+	}
+
+	if (fields & GAIM_GTKCONV_TAB_ICON)
+	{
+		update_tab_icon(conv);
+		generate_send_to_items(win);		/* To update the icons in SendTo menu */
+	}
+
+	if ((fields & GAIM_GTKCONV_TOPIC) &&
+				gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+	{
+		const char *topic;
+		GaimConvChat *chat = GAIM_CONV_CHAT(conv);
+		GaimGtkChatPane *gtkchat = gtkconv->u.chat;
+
+		if (gtkchat->topic_text != NULL)
+		{
+			topic = gaim_conv_chat_get_topic(chat);
+
+			gtk_entry_set_text(GTK_ENTRY(gtkchat->topic_text), topic ? topic : "");
+			gtk_tooltips_set_tip(gtkconv->tooltips, gtkchat->topic_text,
+			                     topic ? topic : "", NULL);
+		}
+	}
+
+	if (fields & GAIM_GTKCONV_SMILEY_THEME)
+		gaim_gtkthemes_smiley_themeize(GAIM_GTK_CONVERSATION(conv)->imhtml);
+
+	if ((fields & GAIM_GTKCONV_COLORIZE_TITLE) ||
+			(fields & GAIM_GTKCONV_SET_TITLE))
+	{
+		char *title;
+		GaimConvIm *im = NULL;
+		GaimAccount *account = gaim_conversation_get_account(conv);
+		AtkObject *accessibility_obj;
+		/* I think this is a little longer than it needs to be but I'm lazy. */
+		char style[51];
+
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+			im = GAIM_CONV_IM(conv);
+
+		if ((account == NULL) ||
+			!gaim_account_is_connected(account) ||
+			((gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_CHAT)
+				&& gaim_conv_chat_has_left(GAIM_CONV_CHAT(conv))))
+			title = g_strdup_printf("(%s)", gaim_conversation_get_title(conv));
+		else
+			title = g_strdup(gaim_conversation_get_title(conv));
+
+		*style = '\0';
+
+		if (!GTK_WIDGET_REALIZED(gtkconv->tab_label))
+			gtk_widget_realize(gtkconv->tab_label);
+
+		accessibility_obj = gtk_widget_get_accessible(gtkconv->tab_cont);
+		if (im != NULL &&
+		    gaim_conv_im_get_typing_state(im) == GAIM_TYPING)
+		{
+			atk_object_set_description(accessibility_obj, _("Typing"));
+			strncpy(style, "color=\"#47A046\"", sizeof(style));
+		}
+		else if (im != NULL &&
+		         gaim_conv_im_get_typing_state(im) == GAIM_TYPED)
+		{
+			atk_object_set_description(accessibility_obj, _("Stopped Typing"));
+			strncpy(style, "color=\"#D1940C\"", sizeof(style));
+		}
+		else if (gtkconv->unseen_state == GAIM_UNSEEN_NICK)
+		{
+			atk_object_set_description(accessibility_obj, _("Nick Said"));
+			strncpy(style, "color=\"#0D4E91\" style=\"italic\" weight=\"bold\"", sizeof(style));
+		}
+		else if (gtkconv->unseen_state == GAIM_UNSEEN_TEXT)
+		{
+			atk_object_set_description(accessibility_obj, _("Unread Messages"));
+			strncpy(style, "color=\"#DF421E\" weight=\"bold\"", sizeof(style));
+		}
+		else if (gtkconv->unseen_state == GAIM_UNSEEN_EVENT)
+		{
+			atk_object_set_description(accessibility_obj, _("New Event"));
+			strncpy(style, "color=\"#868272\" style=\"italic\"", sizeof(style));
+		}
+
+		if (*style != '\0')
+		{
+			char *html_title,*label;
+
+			html_title = g_markup_escape_text(title, -1);
+
+			label = g_strdup_printf("<span %s>%s</span>",
+			                        style, html_title);
+			g_free(html_title);
+			gtk_label_set_markup(GTK_LABEL(gtkconv->tab_label), label);
+			g_free(label);
+		}
+		else
+			gtk_label_set_text(GTK_LABEL(gtkconv->tab_label), title);
+
+		if (gaim_gtk_conv_window_is_active_conversation(conv))
+			update_typing_icon(gtkconv);
+
+		gtk_label_set_text(GTK_LABEL(gtkconv->menu_label), title);
+		if (gaim_gtk_conv_window_is_active_conversation(conv))
+			gtk_window_set_title(GTK_WINDOW(win->window), title);
+
+		g_free(title);
+	}
+}
+
+static void
+gaim_gtkconv_updated(GaimConversation *conv, GaimConvUpdateType type)
+{
+	GaimGtkConvFields flags = 0;
+
+	g_return_if_fail(conv != NULL);
+
+	if (type == GAIM_CONV_UPDATE_ACCOUNT)
+	{
+		flags = GAIM_GTKCONV_ALL;
+	}
+	else if (type == GAIM_CONV_UPDATE_TYPING ||
+	         type == GAIM_CONV_UPDATE_UNSEEN ||
+	         type == GAIM_CONV_UPDATE_TITLE)
+	{
+		flags = GAIM_GTKCONV_COLORIZE_TITLE;
+	}
+	else if (type == GAIM_CONV_UPDATE_TOPIC)
+	{
+		flags = GAIM_GTKCONV_TOPIC;
+	}
+	else if (type == GAIM_CONV_ACCOUNT_ONLINE ||
+	         type == GAIM_CONV_ACCOUNT_OFFLINE)
+	{
+		flags = GAIM_GTKCONV_MENU | GAIM_GTKCONV_TAB_ICON | GAIM_GTKCONV_SET_TITLE;
+	}
+	else if (type == GAIM_CONV_UPDATE_AWAY)
+	{
+		flags = GAIM_GTKCONV_TAB_ICON;
+	}
+	else if (type == GAIM_CONV_UPDATE_ADD ||
+	         type == GAIM_CONV_UPDATE_REMOVE ||
+	         type == GAIM_CONV_UPDATE_CHATLEFT)
+	{
+		flags = GAIM_GTKCONV_SET_TITLE | GAIM_GTKCONV_MENU;
+	}
+	else if (type == GAIM_CONV_UPDATE_ICON)
+	{
+		flags = GAIM_GTKCONV_BUDDY_ICON;
+	}
+	else if (type == GAIM_CONV_UPDATE_FEATURES)
+	{
+		flags = GAIM_GTKCONV_MENU;
+	}
+
+	gaim_gtkconv_update_fields(conv, flags);
+}
+
+static GaimConversationUiOps conversation_ui_ops =
+{
+	gaim_gtkconv_new,
+	gaim_gtkconv_destroy,              /* destroy_conversation */
+	NULL,                              /* write_chat           */
+	gaim_gtkconv_write_im,             /* write_im             */
+	gaim_gtkconv_write_conv,           /* write_conv           */
+	gaim_gtkconv_chat_add_users,       /* chat_add_users       */
+	gaim_gtkconv_chat_rename_user,     /* chat_rename_user     */
+	gaim_gtkconv_chat_remove_users,    /* chat_remove_users    */
+	gaim_gtkconv_chat_update_user,     /* chat_update_user     */
+	gaim_gtkconv_present_conversation, /* present              */
+	gaim_gtkconv_has_focus,            /* has_focus            */
+	gaim_gtkconv_custom_smiley_add,    /* custom_smiley_add    */
+	gaim_gtkconv_custom_smiley_write,  /* custom_smiley_write  */
+	gaim_gtkconv_custom_smiley_close,  /* custom_smiley_close  */
+	gaim_gtkconv_send_confirm,         /* send_confirm         */
+};
+
+GaimConversationUiOps *
+gaim_gtk_conversations_get_conv_ui_ops(void)
+{
+	return &conversation_ui_ops;
+}
+
+/**************************************************************************
+ * Public conversation utility functions
+ **************************************************************************/
+void
+gaim_gtkconv_update_buddy_icon(GaimConversation *conv)
+{
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	GdkPixbufLoader *loader;
+	GdkPixbufAnimation *anim;
+	GError *err = NULL;
+
+	const char *custom = NULL;
+	const void *data = NULL;
+	size_t len;
+
+	GdkPixbuf *buf;
+
+	GtkWidget *event;
+	GtkWidget *frame;
+	GdkPixbuf *scale;
+	int scale_width, scale_height;
+
+	GaimAccount *account;
+	GaimPluginProtocolInfo *prpl_info = NULL;
+
+	GaimBuddyIcon *icon;
+
+	g_return_if_fail(conv != NULL);
+	g_return_if_fail(GAIM_IS_GTK_CONVERSATION(conv));
+	g_return_if_fail(gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM);
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	win = gtkconv->win;
+	if (conv != gtkconv->active_conv)
+		return;
+
+	if (!gtkconv->u.im->show_icon)
+		return;
+
+	account = gaim_conversation_get_account(conv);
+	if(account && account->gc)
+		prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(account->gc->prpl);
+
+	/* Remove the current icon stuff */
+	if (gtkconv->u.im->icon_container != NULL)
+		gtk_widget_destroy(gtkconv->u.im->icon_container);
+	gtkconv->u.im->icon_container = NULL;
+
+	if (gtkconv->u.im->anim != NULL)
+		g_object_unref(G_OBJECT(gtkconv->u.im->anim));
+
+	gtkconv->u.im->anim = NULL;
+
+	if (gtkconv->u.im->icon_timer != 0)
+		g_source_remove(gtkconv->u.im->icon_timer);
+
+	gtkconv->u.im->icon_timer = 0;
+
+	if (gtkconv->u.im->iter != NULL)
+		g_object_unref(G_OBJECT(gtkconv->u.im->iter));
+
+	gtkconv->u.im->iter = NULL;
+
+	if (!gaim_prefs_get_bool("/gaim/gtk/conversations/im/show_buddy_icons"))
+		return;
+
+	if (gaim_conversation_get_gc(conv) == NULL)
+		return;
+
+	custom = custom_icon_pref_name(gtkconv);
+	if (custom) {
+		/* There is a custom icon for this user */
+		char *contents = NULL;
+		if (!g_file_get_contents(custom, &contents, &len, &err)) {
+			gaim_debug_warning("custom icon", "could not load custom icon %s for %s\n",
+						custom, gaim_conversation_get_name(conv));
+			g_error_free(err);
+			err = NULL;
+		} else
+			data = contents;
+	}
+
+	if (data == NULL) {
+		icon = gaim_conv_im_get_icon(GAIM_CONV_IM(conv));
+
+		if (icon == NULL)
+			return;
+
+		data = gaim_buddy_icon_get_data(icon, &len);
+		custom = NULL;
+	}
+
+	loader = gdk_pixbuf_loader_new();
+	gdk_pixbuf_loader_write(loader, data, len, NULL);
+	gdk_pixbuf_loader_close(loader, &err);
+	anim = gdk_pixbuf_loader_get_animation(loader);
+	if (anim)
+		g_object_ref(G_OBJECT(anim));
+	g_object_unref(loader);
+
+	if (custom)
+		g_free((void*)data);
+
+	if (!anim)
+		return;
+	gtkconv->u.im->anim = anim;
+
+	if (err) {
+		gaim_debug(GAIM_DEBUG_ERROR, "gtkconv",
+				   "Buddy icon error: %s\n", err->message);
+		g_error_free(err);
+	}
+
+	if (!gtkconv->u.im->anim)
+		return;
+
+	if (gdk_pixbuf_animation_is_static_image(gtkconv->u.im->anim)) {
+		gtkconv->u.im->iter = NULL;
+		buf = gdk_pixbuf_animation_get_static_image(gtkconv->u.im->anim);
+	} else {
+		gtkconv->u.im->iter =
+			gdk_pixbuf_animation_get_iter(gtkconv->u.im->anim, NULL); /* LEAK */
+		buf = gdk_pixbuf_animation_iter_get_pixbuf(gtkconv->u.im->iter);
+		if (gtkconv->u.im->animate)
+			start_anim(NULL, gtkconv);
+	}
+
+	gaim_gtk_buddy_icon_get_scale_size(buf, &prpl_info->icon_spec,
+			GAIM_ICON_SCALE_DISPLAY, &scale_width, &scale_height);
+	scale = gdk_pixbuf_scale_simple(buf,
+				MAX(gdk_pixbuf_get_width(buf) * scale_width /
+				    gdk_pixbuf_animation_get_width(gtkconv->u.im->anim), 1),
+				MAX(gdk_pixbuf_get_height(buf) * scale_height /
+				    gdk_pixbuf_animation_get_height(gtkconv->u.im->anim), 1),
+				GDK_INTERP_BILINEAR);
+
+	gtkconv->u.im->icon_container = gtk_vbox_new(FALSE, 0);
+
+	frame = gtk_frame_new(NULL);
+	gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_NONE);
+	gtk_box_pack_start(GTK_BOX(gtkconv->u.im->icon_container), frame,
+					   FALSE, FALSE, 0);
+
+	event = gtk_event_box_new();
+	gtk_container_add(GTK_CONTAINER(frame), event);
+	g_signal_connect(G_OBJECT(event), "button-press-event",
+					 G_CALLBACK(icon_menu), gtkconv);
+	gtk_widget_show(event);
+
+	gtkconv->u.im->icon = gtk_image_new_from_pixbuf(scale);
+	gtkconv->auto_resize = TRUE;
+	/* Reset the size request to allow the buddy icon to resize */
+	gtk_widget_set_size_request(gtkconv->lower_hbox, -1, -1);
+	g_idle_add(reset_auto_resize_cb, gtkconv);
+	gtk_widget_set_size_request(gtkconv->u.im->icon, scale_width, scale_height);
+	gtk_container_add(GTK_CONTAINER(event), gtkconv->u.im->icon);
+	gtk_widget_show(gtkconv->u.im->icon);
+
+	g_object_unref(G_OBJECT(scale));
+
+	gtk_box_pack_start(GTK_BOX(gtkconv->lower_hbox),
+			   gtkconv->u.im->icon_container, FALSE, FALSE, 0);
+
+	gtk_widget_show(gtkconv->u.im->icon_container);
+	gtk_widget_show(frame);
+
+	/* The buddy icon code needs badly to be fixed. */
+	if(gaim_gtk_conv_window_is_active_conversation(conv))
+	{
+		buf = gdk_pixbuf_animation_get_static_image(gtkconv->u.im->anim);
+		gtk_window_set_icon(GTK_WINDOW(win->window), buf);
+	}
+}
+
+void
+gaim_gtkconv_update_buttons_by_protocol(GaimConversation *conv)
+{
+	GaimGtkWindow *win;
+
+	if (!GAIM_IS_GTK_CONVERSATION(conv))
+		return;
+
+	win = GAIM_GTK_CONVERSATION(conv)->win;
+
+	if (win != NULL && gaim_gtk_conv_window_is_active_conversation(conv))
+		gray_stuff_out(GAIM_GTK_CONVERSATION(conv));
+}
+
+int
+gaim_gtkconv_get_tab_at_xy(GaimGtkWindow *win, int x, int y, gboolean *to_right)
+{
+	gint nb_x, nb_y, x_rel, y_rel;
+	GtkNotebook *notebook;
+	GtkWidget *page, *tab;
+	gint i, page_num = -1;
+	gint count;
+	gboolean horiz;
+
+	if (to_right)
+		*to_right = FALSE;
+
+	notebook = GTK_NOTEBOOK(win->notebook);
+
+	gdk_window_get_origin(win->notebook->window, &nb_x, &nb_y);
+	x_rel = x - nb_x;
+	y_rel = y - nb_y;
+
+	horiz = (gtk_notebook_get_tab_pos(notebook) == GTK_POS_TOP ||
+			gtk_notebook_get_tab_pos(notebook) == GTK_POS_BOTTOM);
+
+#if GTK_CHECK_VERSION(2,2,0)
+	count = gtk_notebook_get_n_pages(GTK_NOTEBOOK(notebook));
+#else
+	/* this is hacky, but it's only for Gtk 2.0.0... */
+	count = g_list_length(GTK_NOTEBOOK(notebook)->children);
+#endif
+
+	for (i = 0; i < count; i++) {
+
+		page = gtk_notebook_get_nth_page(GTK_NOTEBOOK(notebook), i);
+		tab = gtk_notebook_get_tab_label(GTK_NOTEBOOK(notebook), page);
+
+		/* Make sure the tab is not hidden beyond an arrow */
+		if (!GTK_WIDGET_DRAWABLE(tab))
+			continue;
+
+		if (horiz) {
+			if (x_rel >= tab->allocation.x - GAIM_HIG_BOX_SPACE &&
+					x_rel <= tab->allocation.x + tab->allocation.width + GAIM_HIG_BOX_SPACE) {
+				page_num = i;
+
+				if (to_right && x_rel >= tab->allocation.x + tab->allocation.width/2)
+					*to_right = TRUE;
+
+				break;
+			}
+		} else {
+			if (y_rel >= tab->allocation.y - GAIM_HIG_BOX_SPACE &&
+					y_rel <= tab->allocation.y + tab->allocation.height + GAIM_HIG_BOX_SPACE) {
+				page_num = i;
+
+				if (to_right && y_rel >= tab->allocation.y + tab->allocation.height/2)
+					*to_right = TRUE;
+
+				break;
+			}
+		}
+	}
+
+	if (page_num == -1) {
+		/* Add after the last tab */
+		page_num = count - 1;
+	}
+
+	return page_num;
+}
+
+static void
+close_on_tabs_pref_cb(const char *name, GaimPrefType type,
+					  gconstpointer value, gpointer data)
+{
+	GList *l;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+
+	for (l = gaim_get_conversations(); l != NULL; l = l->next) {
+		conv = (GaimConversation *)l->data;
+
+		if (!GAIM_IS_GTK_CONVERSATION(conv))
+			continue;
+
+		gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+		if (value)
+			gtk_widget_show(gtkconv->close);
+		else
+			gtk_widget_hide(gtkconv->close);
+	}
+}
+
+static void
+spellcheck_pref_cb(const char *name, GaimPrefType type,
+				   gconstpointer value, gpointer data)
+{
+#ifdef USE_GTKSPELL
+	GList *cl;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	GtkSpell *spell;
+
+	for (cl = gaim_get_conversations(); cl != NULL; cl = cl->next) {
+
+		conv = (GaimConversation *)cl->data;
+
+		if (!GAIM_IS_GTK_CONVERSATION(conv))
+			continue;
+
+		gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+		if (value)
+			gaim_gtk_setup_gtkspell(GTK_TEXT_VIEW(gtkconv->entry));
+		else {
+			spell = gtkspell_get_from_text_view(GTK_TEXT_VIEW(gtkconv->entry));
+			gtkspell_detach(spell);
+		}
+	}
+#endif
+}
+
+static void
+tab_side_pref_cb(const char *name, GaimPrefType type,
+				 gconstpointer value, gpointer data)
+{
+	GList *l;
+	GtkPositionType pos;
+	GaimGtkWindow *win;
+
+	pos = GPOINTER_TO_INT(value);
+
+	for (l = gaim_gtk_conv_windows_get_list(); l != NULL; l = l->next) {
+		win = l->data;
+
+		gtk_notebook_set_tab_pos(GTK_NOTEBOOK(win->notebook), pos&~8);
+	}
+}
+
+static void
+show_timestamps_pref_cb(const char *name, GaimPrefType type,
+						gconstpointer value, gpointer data)
+{
+	GList *l;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	for (l = gaim_get_conversations(); l != NULL; l = l->next)
+	{
+		conv = (GaimConversation *)l->data;
+
+		if (!GAIM_IS_GTK_CONVERSATION(conv))
+			continue;
+
+		gtkconv = GAIM_GTK_CONVERSATION(conv);
+		win     = gtkconv->win;
+
+		gtk_check_menu_item_set_active(
+		        GTK_CHECK_MENU_ITEM(win->menu.show_timestamps),
+		        (gboolean)GPOINTER_TO_INT(value));
+
+		gtk_imhtml_show_comments(GTK_IMHTML(gtkconv->imhtml),
+			(gboolean)GPOINTER_TO_INT(value));
+	}
+}
+
+static void
+show_formatting_toolbar_pref_cb(const char *name, GaimPrefType type,
+								gconstpointer value, gpointer data)
+{
+	GList *l;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	for (l = gaim_get_conversations(); l != NULL; l = l->next)
+	{
+		conv = (GaimConversation *)l->data;
+
+		if (!GAIM_IS_GTK_CONVERSATION(conv))
+			continue;
+
+		gtkconv = GAIM_GTK_CONVERSATION(conv);
+		win     = gtkconv->win;
+
+		gtk_check_menu_item_set_active(
+		        GTK_CHECK_MENU_ITEM(win->menu.show_formatting_toolbar),
+		        (gboolean)GPOINTER_TO_INT(value));
+
+		if ((gboolean)GPOINTER_TO_INT(value))
+			gtk_widget_show(gtkconv->toolbar);
+		else
+			gtk_widget_hide(gtkconv->toolbar);
+	}
+}
+
+static void
+animate_buddy_icons_pref_cb(const char *name, GaimPrefType type,
+							gconstpointer value, gpointer data)
+{
+	GList *l;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	if (!gaim_prefs_get_bool("/gaim/gtk/conversations/im/show_buddy_icons"))
+		return;
+
+	/* Set the "animate" flag for each icon based on the new preference */
+	for (l = gaim_get_ims(); l != NULL; l = l->next) {
+		conv = (GaimConversation *)l->data;
+		gtkconv = GAIM_GTK_CONVERSATION(conv);
+		gtkconv->u.im->animate = GPOINTER_TO_INT(value);
+	}
+
+	/* Now either stop or start animation for the active conversation in each window */
+	for (l = gaim_gtk_conv_windows_get_list(); l != NULL; l = l->next) {
+		win = l->data;
+		conv = gaim_gtk_conv_window_get_active_conversation(win);
+		gaim_gtkconv_update_buddy_icon(conv);
+	}
+}
+
+static void
+show_buddy_icons_pref_cb(const char *name, GaimPrefType type,
+						 gconstpointer value, gpointer data)
+{
+	GList *l;
+
+	for (l = gaim_get_conversations(); l != NULL; l = l->next) {
+		GaimConversation *conv = l->data;
+
+		if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+			gaim_gtkconv_update_buddy_icon(conv);
+	}
+}
+
+static void
+conv_placement_usetabs_cb(const char *name, GaimPrefType type,
+						  gconstpointer value, gpointer data)
+{
+	gaim_prefs_trigger_callback("/gaim/gtk/conversations/placement");
+}
+
+static void
+account_status_changed_cb(GaimAccount *account, GaimStatus *oldstatus,
+                          GaimStatus *newstatus)
+{
+	GList *l;
+	GaimConversation *conv = NULL;
+	GaimGtkConversation *gtkconv;
+
+	if(strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/im/hide_new"), "away")!=0)
+		return;
+
+	if(gaim_status_is_available(oldstatus) || !gaim_status_is_available(newstatus))
+		return;
+
+	while ((l = hidden_convwin->gtkconvs) != NULL)
+	{
+		gtkconv = l->data;
+
+		conv = gtkconv->active_conv;
+
+		while(l && !gaim_status_is_available(
+					gaim_account_get_active_status(
+					gaim_conversation_get_account(conv))))
+			l = l->next;
+		if (!l)
+			break;
+
+		gaim_gtk_conv_window_remove_gtkconv(hidden_convwin, gtkconv);
+		gaim_gtkconv_placement_place(gtkconv);
+
+		/* TODO: do we need to do anything for any other conversations that are in the same gtkconv here?
+		 * I'm a little concerned that not doing so will cause the "pending" indicator in the gtkblist not to be cleared. -DAA*/
+		gaim_conversation_update(conv, GAIM_CONV_UPDATE_UNSEEN);
+	}
+}
+
+static void
+hide_new_pref_cb(const char *name, GaimPrefType type,
+				 gconstpointer value, gpointer data)
+{
+	GList *l;
+	GaimConversation *conv = NULL;
+	GaimGtkConversation *gtkconv;
+	gboolean when_away = FALSE;
+
+	if(!hidden_convwin)
+		return;
+
+	if(strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/im/hide_new"), "always")==0)
+		return;
+
+	if(strcmp(gaim_prefs_get_string("/gaim/gtk/conversations/im/hide_new"), "away")==0)
+		when_away = TRUE;
+
+	while ((l = hidden_convwin->gtkconvs) != NULL)
+	{
+		gtkconv = l->data;
+
+		conv = gtkconv->active_conv;
+
+		if(when_away && !gaim_status_is_available(
+							gaim_account_get_active_status(
+							gaim_conversation_get_account(conv))))
+			continue;
+
+		gaim_gtk_conv_window_remove_gtkconv(hidden_convwin, gtkconv);
+		gaim_gtkconv_placement_place(gtkconv);
+	}
+}
+
+
+static void
+conv_placement_pref_cb(const char *name, GaimPrefType type,
+					   gconstpointer value, gpointer data)
+{
+	GaimConvPlacementFunc func;
+
+	if (strcmp(name, "/gaim/gtk/conversations/placement"))
+		return;
+
+	func = gaim_gtkconv_placement_get_fnc(value);
+
+	if (func == NULL)
+		return;
+
+	gaim_gtkconv_placement_set_current_func(func);
+}
+
+static GaimGtkConversation *
+get_gtkconv_with_contact(GaimContact *contact)
+{
+	GaimBlistNode *node;
+
+	node = ((GaimBlistNode*)contact)->child;
+
+	for (; node; node = node->next)
+	{
+		GaimBuddy *buddy = (GaimBuddy*)node;
+		GaimConversation *conv;
+		conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, buddy->name, buddy->account);
+		if (conv)
+			return GAIM_GTK_CONVERSATION(conv);
+	}
+	return NULL;
+}
+
+static void
+account_signed_off_cb(GaimConnection *gc, gpointer event)
+{
+	GList *iter;
+
+	for (iter = gaim_get_conversations(); iter; iter = iter->next)
+	{
+		GaimConversation *conv = iter->data;
+
+		/* This seems fine in theory, but we also need to cover the
+		 * case of this account matching one of the other buddies in
+		 * one of the contacts containing the buddy corresponding to
+		 * a conversation.  It's easier to just update them all. */
+		/* if (gaim_conversation_get_account(conv) == account) */
+			gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON |
+							GAIM_GTKCONV_MENU | GAIM_GTKCONV_COLORIZE_TITLE);
+	}
+}
+
+static gboolean
+update_buddy_status_timeout(GaimBuddy *buddy)
+{
+	/* To remove the signing-on/off door icon */
+	GaimConversation *conv;
+
+	conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, buddy->name, buddy->account);
+	if (conv)
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON);
+
+	return FALSE;
+}
+
+static void
+update_buddy_status_changed(GaimBuddy *buddy, GaimStatus *old, GaimStatus *newstatus)
+{
+	GaimGtkConversation *gtkconv;
+	GaimConversation *conv;
+
+	gtkconv = get_gtkconv_with_contact(gaim_buddy_get_contact(buddy));
+	if (gtkconv)
+	{
+		conv = gtkconv->active_conv;
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON | GAIM_GTKCONV_COLORIZE_TITLE);
+		if ((gaim_status_is_online(old) ^ gaim_status_is_online(newstatus)) != 0)
+			gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_MENU);
+	}
+
+	/* In case a conversation is started after the buddy has signed-on/off */
+	g_timeout_add(11000, (GSourceFunc)update_buddy_status_timeout, buddy);
+}
+
+static void
+update_buddy_privacy_changed(GaimBuddy *buddy)
+{
+	GaimGtkConversation *gtkconv;
+	GaimConversation *conv;
+
+	gtkconv = get_gtkconv_with_contact(gaim_buddy_get_contact(buddy));
+	if (gtkconv)
+	{
+		conv = gtkconv->active_conv;
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON);
+	}
+}
+
+static void
+update_buddy_idle_changed(GaimBuddy *buddy, gboolean old, gboolean newidle)
+{
+	GaimConversation *conv;
+
+	conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, buddy->name, buddy->account);
+	if (conv)
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON);
+}
+
+static void
+update_buddy_icon(GaimBuddy *buddy)
+{
+	GaimConversation *conv;
+
+	conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, buddy->name, buddy->account);
+	if (conv)
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_BUDDY_ICON);
+}
+
+static void
+update_buddy_sign(GaimBuddy *buddy, const char *which)
+{
+	GaimPresence *presence;
+	GaimStatus *on, *off;
+
+	presence = gaim_buddy_get_presence(buddy);
+	if (!presence)
+		return;
+	off = gaim_presence_get_status(presence, "offline");
+	on = gaim_presence_get_status(presence, "available");
+
+	if (*(which+1) == 'f')
+		update_buddy_status_changed(buddy, on, off);
+	else
+		update_buddy_status_changed(buddy, off, on);
+}
+
+static void
+update_conversation_switched(GaimConversation *conv)
+{
+	gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TAB_ICON | GAIM_GTKCONV_SET_TITLE |
+					GAIM_GTKCONV_MENU | GAIM_GTKCONV_BUDDY_ICON);
+}
+
+static void
+update_buddy_typing(GaimAccount *account, const char *who)
+{
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+
+	conv = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, who, account);
+	if (!conv)
+		return;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+	if (gtkconv && gtkconv->active_conv == conv)
+		gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_COLORIZE_TITLE);
+}
+
+static void
+update_chat(GaimConversation *conv)
+{
+	gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TOPIC |
+					GAIM_GTKCONV_MENU | GAIM_GTKCONV_SET_TITLE);
+}
+
+static void
+update_chat_topic(GaimConversation *conv, const char *old, const char *new)
+{
+	gaim_gtkconv_update_fields(conv, GAIM_GTKCONV_TOPIC);
+}
+
+void *
+gaim_gtk_conversations_get_handle(void)
+{
+	static int handle;
+
+	return &handle;
+}
+
+void
+gaim_gtk_conversations_init(void)
+{
+	void *handle = gaim_gtk_conversations_get_handle();
+	void *blist_handle = gaim_blist_get_handle();
+
+	/* Conversations */
+	gaim_prefs_add_none("/gaim/gtk/conversations");
+	gaim_prefs_add_bool("/gaim/gtk/conversations/use_smooth_scrolling", TRUE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/close_on_tabs", TRUE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/send_bold", FALSE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/send_italic", FALSE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/send_underline", FALSE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/spellcheck", TRUE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/show_incoming_formatting", TRUE);
+
+	gaim_prefs_add_bool("/gaim/gtk/conversations/show_timestamps", TRUE);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/show_formatting_toolbar", TRUE);
+
+	gaim_prefs_add_string("/gaim/gtk/conversations/placement", "last");
+	gaim_prefs_add_int("/gaim/gtk/conversations/placement_number", 1);
+	gaim_prefs_add_string("/gaim/gtk/conversations/bgcolor", "");
+	gaim_prefs_add_string("/gaim/gtk/conversations/fgcolor", "");
+	gaim_prefs_add_string("/gaim/gtk/conversations/font_face", "");
+	gaim_prefs_add_int("/gaim/gtk/conversations/font_size", 3);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/tabs", TRUE);
+	gaim_prefs_add_int("/gaim/gtk/conversations/tab_side", GTK_POS_TOP);
+	gaim_prefs_add_int("/gaim/gtk/conversations/scrollback_lines", 4000);
+
+	/* Conversations -> Chat */
+	gaim_prefs_add_none("/gaim/gtk/conversations/chat");
+	gaim_prefs_add_int("/gaim/gtk/conversations/chat/default_width", 410);
+	gaim_prefs_add_int("/gaim/gtk/conversations/chat/default_height", 160);
+	gaim_prefs_add_int("/gaim/gtk/conversations/chat/entry_height", 50);
+	gaim_prefs_add_int("/gaim/gtk/conversations/chat/userlist_width", 80);
+	/* Conversations -> IM */
+	gaim_prefs_add_none("/gaim/gtk/conversations/im");
+
+	gaim_prefs_add_bool("/gaim/gtk/conversations/im/animate_buddy_icons", TRUE);
+
+	gaim_prefs_add_int("/gaim/gtk/conversations/im/default_width", 410);
+	gaim_prefs_add_int("/gaim/gtk/conversations/im/default_height", 160);
+	gaim_prefs_add_int("/gaim/gtk/conversations/im/entry_height", 50);
+	gaim_prefs_add_bool("/gaim/gtk/conversations/im/show_buddy_icons", TRUE);
+
+	gaim_prefs_add_string("/gaim/gtk/conversations/im/hide_new", "never");
+
+	/* Connect callbacks. */
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/close_on_tabs",
+								close_on_tabs_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/show_timestamps",
+								show_timestamps_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/show_formatting_toolbar",
+								show_formatting_toolbar_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/spellcheck",
+								spellcheck_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/tab_side",
+								tab_side_pref_cb, NULL);
+
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/tabs",
+								conv_placement_usetabs_cb, NULL);
+
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/placement",
+								conv_placement_pref_cb, NULL);
+	gaim_prefs_trigger_callback("/gaim/gtk/conversations/placement");
+
+	/* IM callbacks */
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/im/animate_buddy_icons",
+								animate_buddy_icons_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/im/show_buddy_icons",
+								show_buddy_icons_pref_cb, NULL);
+	gaim_prefs_connect_callback(handle, "/gaim/gtk/conversations/im/hide_new",
+                                hide_new_pref_cb, NULL);
+
+
+
+	/**********************************************************************
+	 * Register signals
+	 **********************************************************************/
+	gaim_signal_register(handle, "conversation-dragging",
+	                     gaim_marshal_VOID__POINTER_POINTER, NULL, 2,
+	                     gaim_value_new(GAIM_TYPE_BOXED,
+	                                    "GaimGtkWindow *"),
+	                     gaim_value_new(GAIM_TYPE_BOXED,
+	                                    "GaimGtkWindow *"));
+
+	gaim_signal_register(handle, "conversation-timestamp",
+#if SIZEOF_TIME_T == 4
+	                     gaim_marshal_POINTER__POINTER_INT,
+#elif SIZEOF_TIME_T == 8
+			     gaim_marshal_POINTER__POINTER_INT64,
+#else
+#error Unkown size of time_t
+#endif
+	                     gaim_value_new(GAIM_TYPE_POINTER), 2,
+	                     gaim_value_new(GAIM_TYPE_SUBTYPE,
+	                                    GAIM_SUBTYPE_CONVERSATION),
+#if SIZEOF_TIME_T == 4
+	                     gaim_value_new(GAIM_TYPE_INT));
+#elif SIZEOF_TIME_T == 8
+	                     gaim_value_new(GAIM_TYPE_INT64));
+#else
+# error Unknown size of time_t
+#endif
+
+	gaim_signal_register(handle, "displaying-im-msg",
+						 gaim_marshal_BOOLEAN__POINTER_POINTER_POINTER_POINTER_POINTER,
+						 gaim_value_new(GAIM_TYPE_BOOLEAN), 5,
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_ACCOUNT),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new_outgoing(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_CONVERSATION),
+						 gaim_value_new(GAIM_TYPE_INT));
+
+	gaim_signal_register(handle, "displayed-im-msg",
+						 gaim_marshal_VOID__POINTER_POINTER_POINTER_POINTER_UINT,
+						 NULL, 5,
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_ACCOUNT),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_CONVERSATION),
+						 gaim_value_new(GAIM_TYPE_INT));
+
+	gaim_signal_register(handle, "displaying-chat-msg",
+						 gaim_marshal_BOOLEAN__POINTER_POINTER_POINTER_POINTER_POINTER,
+						 gaim_value_new(GAIM_TYPE_BOOLEAN), 5,
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_ACCOUNT),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new_outgoing(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_CONVERSATION),
+						 gaim_value_new(GAIM_TYPE_INT));
+
+	gaim_signal_register(handle, "displayed-chat-msg",
+						 gaim_marshal_VOID__POINTER_POINTER_POINTER_POINTER_UINT,
+						 NULL, 5,
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_ACCOUNT),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_STRING),
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_CONVERSATION),
+						 gaim_value_new(GAIM_TYPE_INT));
+
+	gaim_signal_register(handle, "conversation-switched",
+						 gaim_marshal_VOID__POINTER_POINTER, NULL, 1,
+						 gaim_value_new(GAIM_TYPE_SUBTYPE,
+										GAIM_SUBTYPE_CONVERSATION));
+
+	/**********************************************************************
+	 * Register commands
+	 **********************************************************************/
+	gaim_cmd_register("say", "S", GAIM_CMD_P_DEFAULT,
+	                  GAIM_CMD_FLAG_CHAT | GAIM_CMD_FLAG_IM, NULL,
+	                  say_command_cb, _("say &lt;message&gt;:  Send a message normally as if you weren't using a command."), NULL);
+	gaim_cmd_register("me", "S", GAIM_CMD_P_DEFAULT,
+	                  GAIM_CMD_FLAG_CHAT | GAIM_CMD_FLAG_IM, NULL,
+	                  me_command_cb, _("me &lt;action&gt;:  Send an IRC style action to a buddy or chat."), NULL);
+	gaim_cmd_register("debug", "w", GAIM_CMD_P_DEFAULT,
+	                  GAIM_CMD_FLAG_CHAT | GAIM_CMD_FLAG_IM, NULL,
+	                  debug_command_cb, _("debug &lt;option&gt;:  Send various debug information to the current conversation."), NULL);
+	gaim_cmd_register("clear", "", GAIM_CMD_P_DEFAULT,
+	                  GAIM_CMD_FLAG_CHAT | GAIM_CMD_FLAG_IM, NULL,
+	                  clear_command_cb, _("clear: Clears the conversation scrollback."), NULL);
+	gaim_cmd_register("help", "w", GAIM_CMD_P_DEFAULT,
+	                  GAIM_CMD_FLAG_CHAT | GAIM_CMD_FLAG_IM | GAIM_CMD_FLAG_ALLOW_WRONG_ARGS, NULL,
+	                  help_command_cb, _("help &lt;command&gt;:  Help on a specific command."), NULL);
+
+	/**********************************************************************
+	 * UI operations
+	 **********************************************************************/
+
+	gaim_signal_connect(gaim_connections_get_handle(), "signed-on", handle,
+						G_CALLBACK(account_signed_off_cb),
+						GINT_TO_POINTER(GAIM_CONV_ACCOUNT_ONLINE));
+	gaim_signal_connect(gaim_connections_get_handle(), "signed-off", handle,
+						G_CALLBACK(account_signed_off_cb),
+						GINT_TO_POINTER(GAIM_CONV_ACCOUNT_OFFLINE));
+
+	gaim_signal_connect(gaim_conversations_get_handle(), "received-im-msg",
+						handle, G_CALLBACK(received_im_msg_cb), NULL);
+
+	gaim_conversations_set_ui_ops(&conversation_ui_ops);
+
+	hidden_convwin = gaim_gtk_conv_window_new();
+	window_list = g_list_remove(window_list, hidden_convwin);
+
+	gaim_signal_connect(gaim_accounts_get_handle(), "account-status-changed",
+                        handle, GAIM_CALLBACK(account_status_changed_cb), NULL);
+
+	/* Callbacks to update a conversation */
+	gaim_signal_connect(blist_handle, "buddy-added", handle,
+						G_CALLBACK(buddy_update_cb), NULL);
+	gaim_signal_connect(blist_handle, "buddy-removed", handle,
+						G_CALLBACK(buddy_update_cb), NULL);
+	gaim_signal_connect(blist_handle, "buddy-signed-on",
+						handle, GAIM_CALLBACK(update_buddy_sign), "on");
+	gaim_signal_connect(blist_handle, "buddy-signed-off",
+						handle, GAIM_CALLBACK(update_buddy_sign), "off");
+	gaim_signal_connect(blist_handle, "buddy-status-changed",
+						handle, GAIM_CALLBACK(update_buddy_status_changed), NULL);
+	gaim_signal_connect(blist_handle, "buddy-privacy-changed",
+						handle, GAIM_CALLBACK(update_buddy_privacy_changed), NULL);
+	gaim_signal_connect(blist_handle, "buddy-idle-changed",
+						handle, GAIM_CALLBACK(update_buddy_idle_changed), NULL);
+	gaim_signal_connect(blist_handle, "buddy-icon-changed",
+						handle, GAIM_CALLBACK(update_buddy_icon), NULL);
+	gaim_signal_connect(gaim_conversations_get_handle(), "buddy-typing",
+						handle, GAIM_CALLBACK(update_buddy_typing), NULL);
+	gaim_signal_connect(gaim_conversations_get_handle(), "buddy-typing-stopped",
+						handle, GAIM_CALLBACK(update_buddy_typing), NULL);
+	gaim_signal_connect(gaim_gtk_conversations_get_handle(), "conversation-switched",
+						handle, GAIM_CALLBACK(update_conversation_switched), NULL);
+	gaim_signal_connect(gaim_conversations_get_handle(), "chat-left", handle,
+						GAIM_CALLBACK(update_chat), NULL);
+	gaim_signal_connect(gaim_conversations_get_handle(), "chat-joined", handle,
+						GAIM_CALLBACK(update_chat), NULL);
+	gaim_signal_connect(gaim_conversations_get_handle(), "chat-topic-changed", handle,
+						GAIM_CALLBACK(update_chat_topic), NULL);
+	gaim_signal_connect_priority(gaim_conversations_get_handle(), "conversation-updated", handle,
+						GAIM_CALLBACK(gaim_gtkconv_updated), NULL,
+						GAIM_SIGNAL_PRIORITY_LOWEST);
+}
+
+void
+gaim_gtk_conversations_uninit(void)
+{
+	gaim_prefs_disconnect_by_handle(gaim_gtk_conversations_get_handle());
+	gaim_signals_disconnect_by_handle(gaim_gtk_conversations_get_handle());
+	gaim_signals_unregister_by_instance(gaim_gtk_conversations_get_handle());
+	gaim_gtk_conv_window_destroy(hidden_convwin);
+	hidden_convwin=NULL;
+}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+/* down here is where gtkconvwin.c ought to start. except they share like every freaking function,
+ * and touch each others' private members all day long */
+
+/**
+ * @file gtkconvwin.c GTK+ Conversation Window 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 <gdk/gdkkeysyms.h>
+
+#include "account.h"
+#include "cmds.h"
+#include "debug.h"
+#include "imgstore.h"
+#include "log.h"
+#include "notify.h"
+#include "prpl.h"
+#include "request.h"
+#include "util.h"
+
+#include "gtkdnd-hints.h"
+#include "gtkblist.h"
+#include "gtkconv.h"
+#include "gtkdialogs.h"
+#include "gtkmenutray.h"
+#include "gtkpounce.h"
+#include "gtkprefs.h"
+#include "gtkprivacy.h"
+#include "gtkutils.h"
+#include "gaimstock.h"
+#include "gtkimhtml.h"
+#include "gtkimhtmltoolbar.h"
+
+static void
+do_close(GtkWidget *w, int resp, GaimGtkWindow *win)
+{
+	gtk_widget_destroy(warn_close_dialog);
+	warn_close_dialog = NULL;
+
+	if (resp == GTK_RESPONSE_OK)
+		gaim_gtk_conv_window_destroy(win);
+}
+
+static void
+build_warn_close_dialog(GaimGtkWindow *gtkwin)
+{
+	GtkWidget *label;
+	GtkWidget *vbox, *hbox;
+	GtkWidget *img;
+
+	g_return_if_fail(warn_close_dialog == NULL);
+
+
+	warn_close_dialog = gtk_dialog_new_with_buttons(
+							_("Confirm close"),
+							GTK_WINDOW(gtkwin->window), GTK_DIALOG_MODAL,
+							GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
+							GAIM_STOCK_CLOSE_TABS, GTK_RESPONSE_OK, NULL);
+
+	gtk_dialog_set_default_response(GTK_DIALOG(warn_close_dialog),
+	                                GTK_RESPONSE_OK);
+
+	gtk_container_set_border_width(GTK_CONTAINER(warn_close_dialog),
+	                               6);
+	gtk_window_set_resizable(GTK_WINDOW(warn_close_dialog), FALSE);
+	gtk_dialog_set_has_separator(GTK_DIALOG(warn_close_dialog),
+	                             FALSE);
+
+	/* Setup the outside spacing. */
+	vbox = GTK_DIALOG(warn_close_dialog)->vbox;
+
+	gtk_box_set_spacing(GTK_BOX(vbox), 12);
+	gtk_container_set_border_width(GTK_CONTAINER(vbox), 6);
+
+	img = gtk_image_new_from_stock(GAIM_STOCK_DIALOG_WARNING,
+	                               GTK_ICON_SIZE_DIALOG);
+	/* Setup the inner hbox and put the dialog's icon in it. */
+	hbox = gtk_hbox_new(FALSE, 12);
+	gtk_container_add(GTK_CONTAINER(vbox), hbox);
+	gtk_box_pack_start(GTK_BOX(hbox), img, FALSE, FALSE, 0);
+	gtk_misc_set_alignment(GTK_MISC(img), 0, 0);
+
+	/* Setup the right vbox. */
+	vbox = gtk_vbox_new(FALSE, 12);
+	gtk_container_add(GTK_CONTAINER(hbox), vbox);
+
+	label = gtk_label_new(_("You have unread messages. Are you sure you want to close the window?"));
+	gtk_widget_set_size_request(label, 350, -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);
+
+	/* Connect the signals. */
+	g_signal_connect(G_OBJECT(warn_close_dialog), "response",
+	                 G_CALLBACK(do_close), gtkwin);
+
+}
+
+/**************************************************************************
+ * Callbacks
+ **************************************************************************/
+
+static gboolean
+close_win_cb(GtkWidget *w, GdkEventAny *e, gpointer d)
+{
+	GaimGtkWindow *win = d;
+	GList *l;
+
+	/* If there are unread messages then show a warning dialog */
+	for (l = gaim_gtk_conv_window_get_gtkconvs(win);
+	     l != NULL; l = l->next)
+	{
+		GaimGtkConversation *gtkconv = l->data;
+		if (gaim_conversation_get_type(gtkconv->active_conv) == GAIM_CONV_TYPE_IM &&
+				gtkconv->unseen_state >= GAIM_UNSEEN_TEXT)
+		{
+			build_warn_close_dialog(win);
+			gtk_widget_show_all(warn_close_dialog);
+
+			return TRUE;
+		}
+	}
+
+	gaim_gtk_conv_window_destroy(win);
+
+	return TRUE;
+}
+
+static void
+gtkconv_set_unseen(GaimGtkConversation *gtkconv, GaimUnseenState state)
+{
+	if (state == GAIM_UNSEEN_NONE)
+	{
+		gtkconv->unseen_count = 0;
+		gtkconv->unseen_state = GAIM_UNSEEN_NONE;
+	}
+	else
+	{
+		if (state >= GAIM_UNSEEN_TEXT)
+			gtkconv->unseen_count++;
+
+		if (state > gtkconv->unseen_state)
+			gtkconv->unseen_state = state;
+	}
+
+	gaim_conversation_update(gtkconv->active_conv, GAIM_CONV_UPDATE_UNSEEN);
+}
+
+/*
+ * When a conversation window is focused, we know the user
+ * has looked at it so we know there are no longer unseen
+ * messages.
+ */
+static gint
+focus_win_cb(GtkWidget *w, GdkEventFocus *e, gpointer d)
+{
+	GaimGtkWindow *win = d;
+	GaimGtkConversation *gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+
+	gtkconv_set_unseen(gtkconv, GAIM_UNSEEN_NONE);
+
+	return FALSE;
+}
+
+#if !GTK_CHECK_VERSION(2,6,0)
+/* Courtesy of Galeon! */
+static void
+tab_close_button_state_changed_cb(GtkWidget *widget, GtkStateType prev_state)
+{
+	if (GTK_WIDGET_STATE(widget) == GTK_STATE_ACTIVE)
+		gtk_widget_set_state(widget, GTK_STATE_NORMAL);
+}
+#endif
+
+static void
+notebook_init_grab(GaimGtkWindow *gtkwin, GtkWidget *widget)
+{
+	static GdkCursor *cursor = NULL;
+
+	gtkwin->in_drag = TRUE;
+
+	if (gtkwin->drag_leave_signal) {
+		g_signal_handler_disconnect(G_OBJECT(widget),
+		                            gtkwin->drag_leave_signal);
+		gtkwin->drag_leave_signal = 0;
+	}
+
+	if (cursor == NULL)
+		cursor = gdk_cursor_new(GDK_FLEUR);
+
+	/* Grab the pointer */
+	gtk_grab_add(gtkwin->notebook);
+#ifndef _WIN32
+	/* Currently for win32 GTK+ (as of 2.2.1), gdk_pointer_is_grabbed will
+	   always be true after a button press. */
+	if (!gdk_pointer_is_grabbed())
+#endif
+		gdk_pointer_grab(gtkwin->notebook->window, FALSE,
+		                 GDK_BUTTON1_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
+		                 NULL, cursor, GDK_CURRENT_TIME);
+}
+
+static gboolean
+notebook_motion_cb(GtkWidget *widget, GdkEventButton *e, GaimGtkWindow *win)
+{
+
+	/*
+	* Make sure the user moved the mouse far enough for the
+	* drag to be initiated.
+	*/
+	if (win->in_predrag) {
+		if (e->x_root <  win->drag_min_x ||
+		    e->x_root >= win->drag_max_x ||
+		    e->y_root <  win->drag_min_y ||
+		    e->y_root >= win->drag_max_y) {
+
+			    win->in_predrag = FALSE;
+			    notebook_init_grab(win, widget);
+		    }
+	}
+	else { /* Otherwise, draw the arrows. */
+		GaimGtkWindow *dest_win;
+		GtkNotebook *dest_notebook;
+		GtkWidget *tab;
+		gint nb_x, nb_y, page_num;
+		gint arrow1_x, arrow1_y, arrow2_x, arrow2_y;
+		gboolean horiz_tabs = FALSE;
+		GaimGtkConversation *gtkconv;
+		gboolean to_right = FALSE;
+
+		/* Get the window that the cursor is over. */
+		dest_win = gaim_gtk_conv_window_get_at_xy(e->x_root, e->y_root);
+
+		if (dest_win == NULL) {
+			dnd_hints_hide_all();
+
+			return TRUE;
+		}
+
+		dest_notebook = GTK_NOTEBOOK(dest_win->notebook);
+
+		gdk_window_get_origin(GTK_WIDGET(dest_notebook)->window, &nb_x, &nb_y);
+
+		arrow1_x = arrow2_x = nb_x;
+		arrow1_y = arrow2_y = nb_y;
+
+		page_num = gaim_gtkconv_get_tab_at_xy(dest_win,
+		                                      e->x_root, e->y_root, &to_right);
+		to_right = to_right && (win != dest_win);
+
+		if (gtk_notebook_get_tab_pos(dest_notebook) == GTK_POS_TOP ||
+		    gtk_notebook_get_tab_pos(dest_notebook) == GTK_POS_BOTTOM) {
+
+			    horiz_tabs = TRUE;
+		    }
+
+		gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(dest_win, page_num);
+		tab = gtkconv->tabby;
+
+		if (horiz_tabs) {
+			arrow1_x = arrow2_x = nb_x + tab->allocation.x;
+
+			if (((gpointer)win == (gpointer)dest_win && win->drag_tab < page_num) || to_right) {
+				arrow1_x += tab->allocation.width;
+				arrow2_x += tab->allocation.width;
+			}
+
+			arrow1_y = nb_y + tab->allocation.y;
+			arrow2_y = nb_y + tab->allocation.y + tab->allocation.height;
+		} else {
+			arrow1_x = nb_x + tab->allocation.x;
+			arrow2_x = nb_x + tab->allocation.x + tab->allocation.width;
+			arrow1_y = arrow2_y = nb_y + tab->allocation.y;
+
+			if (((gpointer)win == (gpointer)dest_win && win->drag_tab < page_num) || to_right) {
+				arrow1_y += tab->allocation.height;
+				arrow2_y += tab->allocation.height;
+			}
+		}
+
+		if (horiz_tabs) {
+			dnd_hints_show(HINT_ARROW_DOWN, arrow1_x, arrow1_y);
+			dnd_hints_show(HINT_ARROW_UP,   arrow2_x, arrow2_y);
+		} else {
+			dnd_hints_show(HINT_ARROW_RIGHT, arrow1_x, arrow1_y);
+			dnd_hints_show(HINT_ARROW_LEFT,  arrow2_x, arrow2_y);
+		}
+	}
+
+	return TRUE;
+}
+
+static gboolean
+notebook_leave_cb(GtkWidget *widget, GdkEventCrossing *e, GaimGtkWindow *win)
+{
+	if (win->in_drag)
+		return FALSE;
+
+	if (e->x_root <  win->drag_min_x ||
+	    e->x_root >= win->drag_max_x ||
+	    e->y_root <  win->drag_min_y ||
+	    e->y_root >= win->drag_max_y) {
+
+		    win->in_predrag = FALSE;
+		    notebook_init_grab(win, widget);
+	    }
+
+	return TRUE;
+}
+
+/*
+ * THANK YOU GALEON!
+ */
+static gboolean
+notebook_press_cb(GtkWidget *widget, GdkEventButton *e, GaimGtkWindow *win)
+{
+	gint nb_x, nb_y, x_rel, y_rel;
+	int tab_clicked;
+	GtkWidget *page;
+	GtkWidget *tab;
+
+	if (e->button == 2) {
+		GaimGtkConversation *gtkconv;
+		tab_clicked = gaim_gtkconv_get_tab_at_xy(win, e->x_root, e->y_root, NULL);
+
+		if (tab_clicked == -1)
+			return FALSE;
+
+		gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win, tab_clicked);
+		close_conv_cb(NULL, gtkconv);
+		return TRUE;
+	}
+
+
+	if (e->button != 1 || e->type != GDK_BUTTON_PRESS)
+		return FALSE;
+
+
+	if (win->in_drag) {
+		gaim_debug(GAIM_DEBUG_WARNING, "gtkconv",
+		           "Already in the middle of a window drag at tab_press_cb\n");
+		return TRUE;
+	}
+
+	/*
+	* Make sure a tab was actually clicked. The arrow buttons
+	* mess things up.
+	*/
+	tab_clicked = gaim_gtkconv_get_tab_at_xy(win, e->x_root, e->y_root, NULL);
+
+	if (tab_clicked == -1)
+		return FALSE;
+
+	/*
+	* Get the relative position of the press event, with regards to
+	* the position of the notebook.
+	*/
+	gdk_window_get_origin(win->notebook->window, &nb_x, &nb_y);
+
+	x_rel = e->x_root - nb_x;
+	y_rel = e->y_root - nb_y;
+
+	/* Reset the min/max x/y */
+	win->drag_min_x = 0;
+	win->drag_min_y = 0;
+	win->drag_max_x = 0;
+	win->drag_max_y = 0;
+
+	/* Find out which tab was dragged. */
+	page = gtk_notebook_get_nth_page(GTK_NOTEBOOK(win->notebook), tab_clicked);
+	tab = gtk_notebook_get_tab_label(GTK_NOTEBOOK(win->notebook), page);
+
+	win->drag_min_x = tab->allocation.x      + nb_x;
+	win->drag_min_y = tab->allocation.y      + nb_y;
+	win->drag_max_x = tab->allocation.width  + win->drag_min_x;
+	win->drag_max_y = tab->allocation.height + win->drag_min_y;
+
+	/* Make sure the click occurred in the tab. */
+	if (e->x_root <  win->drag_min_x ||
+	    e->x_root >= win->drag_max_x ||
+	    e->y_root <  win->drag_min_y ||
+	    e->y_root >= win->drag_max_y) {
+
+		    return FALSE;
+	    }
+
+	win->in_predrag = TRUE;
+	win->drag_tab = tab_clicked;
+
+	/* Connect the new motion signals. */
+	win->drag_motion_signal =
+		g_signal_connect(G_OBJECT(widget), "motion_notify_event",
+		                 G_CALLBACK(notebook_motion_cb), win);
+
+	win->drag_leave_signal =
+		g_signal_connect(G_OBJECT(widget), "leave_notify_event",
+		                 G_CALLBACK(notebook_leave_cb), win);
+
+	return FALSE;
+}
+
+static gboolean
+notebook_release_cb(GtkWidget *widget, GdkEventButton *e, GaimGtkWindow *win)
+{
+	GaimGtkWindow *dest_win;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	gint dest_page_num = 0;
+	gboolean new_window = FALSE;
+	gboolean to_right = FALSE;
+
+	/*
+	* Don't check to make sure that the event's window matches the
+	* widget's, because we may be getting an event passed on from the
+	* close button.
+	*/
+	if (e->button != 1 && e->type != GDK_BUTTON_RELEASE)
+		return FALSE;
+
+	if (gdk_pointer_is_grabbed()) {
+		gdk_pointer_ungrab(GDK_CURRENT_TIME);
+		gtk_grab_remove(widget);
+	}
+
+	if (!win->in_predrag && !win->in_drag)
+		return FALSE;
+
+	/* Disconnect the motion signal. */
+	if (win->drag_motion_signal) {
+		g_signal_handler_disconnect(G_OBJECT(widget),
+		                            win->drag_motion_signal);
+
+		win->drag_motion_signal = 0;
+	}
+
+	/*
+	* If we're in a pre-drag, we'll also need to disconnect the leave
+	* signal.
+	*/
+	if (win->in_predrag) {
+		win->in_predrag = FALSE;
+
+		if (win->drag_leave_signal) {
+			g_signal_handler_disconnect(G_OBJECT(widget),
+			                            win->drag_leave_signal);
+
+			win->drag_leave_signal = 0;
+		}
+	}
+
+	/* If we're not in drag...        */
+	/* We're perfectly normal people! */
+	if (!win->in_drag)
+		return FALSE;
+
+	win->in_drag = FALSE;
+
+	dnd_hints_hide_all();
+
+	dest_win = gaim_gtk_conv_window_get_at_xy(e->x_root, e->y_root);
+
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	if (dest_win == NULL) {
+		/* If the current window doesn't have any other conversations,
+		* there isn't much point transferring the conv to a new window. */
+		if (gaim_gtk_conv_window_get_gtkconv_count(win) > 1) {
+			/* Make a new window to stick this to. */
+			dest_win = gaim_gtk_conv_window_new();
+			new_window = TRUE;
+		}
+	}
+
+	if (dest_win == NULL)
+		return FALSE;
+
+	gaim_signal_emit(gaim_gtk_conversations_get_handle(),
+	                 "conversation-dragging", win, dest_win);
+
+	/* Get the destination page number. */
+	if (!new_window)
+		dest_page_num = gaim_gtkconv_get_tab_at_xy(dest_win,
+		                                           e->x_root, e->y_root, &to_right);
+
+	gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win, win->drag_tab);
+
+	if (win == dest_win) {
+		gtk_notebook_reorder_child(GTK_NOTEBOOK(win->notebook), gtkconv->tab_cont, dest_page_num);
+	} else {
+		gaim_gtk_conv_window_remove_gtkconv(win, gtkconv);
+		gaim_gtk_conv_window_add_gtkconv(dest_win, gtkconv);
+		gtk_notebook_reorder_child(GTK_NOTEBOOK(dest_win->notebook), gtkconv->tab_cont, dest_page_num + to_right);
+		gaim_gtk_conv_window_switch_gtkconv(dest_win, gtkconv);
+		if (new_window) {
+			gint win_width, win_height;
+
+			gtk_window_get_size(GTK_WINDOW(dest_win->window),
+			                    &win_width, &win_height);
+
+			gtk_window_move(GTK_WINDOW(dest_win->window),
+			                e->x_root - (win_width  / 2),
+			                e->y_root - (win_height / 2));
+
+			gaim_gtk_conv_window_show(dest_win);
+		}
+	}
+
+	gtk_widget_grab_focus(GAIM_GTK_CONVERSATION(conv)->entry);
+
+	return TRUE;
+}
+
+
+static void
+before_switch_conv_cb(GtkNotebook *notebook, GtkWidget *page, gint page_num,
+                      gpointer user_data)
+{
+	GaimGtkWindow *win;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+
+	win = user_data;
+	conv = gaim_gtk_conv_window_get_active_conversation(win);
+
+	g_return_if_fail(conv != NULL);
+
+	if (gaim_conversation_get_type(conv) != GAIM_CONV_TYPE_IM)
+		return;
+
+	gtkconv = GAIM_GTK_CONVERSATION(conv);
+
+	stop_anim(NULL, gtkconv);
+}
+static void
+close_window(GtkWidget *w, GaimGtkWindow *win)
+{
+	close_win_cb(w, NULL, win);
+}
+
+static void
+detach_tab_cb(GtkWidget *w, GObject *menu)
+{
+	GaimGtkWindow *win, *new_window;
+	GaimGtkConversation *gtkconv;
+
+	gtkconv = g_object_get_data(menu, "clicked_tab");
+
+	if (!gtkconv)
+		return;
+
+	win = gaim_gtkconv_get_window(gtkconv);
+	/* Nothing to do if there's only one tab in the window */
+	if (gaim_gtk_conv_window_get_gtkconv_count(win) == 1)
+		return;
+
+	gaim_gtk_conv_window_remove_gtkconv(win, gtkconv);
+
+	new_window = gaim_gtk_conv_window_new();
+	gaim_gtk_conv_window_add_gtkconv(new_window, gtkconv);
+	gaim_gtk_conv_window_show(new_window);
+}
+
+static void
+close_others_cb(GtkWidget *w, GObject *menu)
+{
+	GList *iter;
+	GaimGtkConversation *gtkconv;
+	GaimGtkWindow *win;
+
+	gtkconv = g_object_get_data(menu, "clicked_tab");
+
+	if (!gtkconv)
+		return;
+
+	win = gaim_gtkconv_get_window(gtkconv);
+
+	for (iter = gaim_gtk_conv_window_get_gtkconvs(win); iter; )
+	{
+		GaimGtkConversation *gconv = iter->data;
+		iter = iter->next;
+
+		if (gconv != gtkconv)
+		{
+			close_conv_cb(NULL, gconv);
+		}
+	}
+}
+
+static void close_tab_cb(GtkWidget *w, GObject *menu)
+{
+	GaimGtkConversation *gtkconv;
+
+	gtkconv = g_object_get_data(menu, "clicked_tab");
+
+	if (gtkconv)
+		close_conv_cb(NULL, gtkconv);
+}
+
+static gboolean
+right_click_menu_cb(GtkNotebook *notebook, GdkEventButton *event, GaimGtkWindow *win)
+{
+	GtkWidget *item, *menu;
+	GaimGtkConversation *gtkconv;
+
+	if (event->type != GDK_BUTTON_PRESS || event->button != 3)
+		return FALSE;
+
+	gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win,
+			gaim_gtkconv_get_tab_at_xy(win, event->x_root, event->y_root, NULL));
+
+	if (g_object_get_data(G_OBJECT(notebook->menu), "clicked_tab"))
+	{
+		g_object_set_data(G_OBJECT(notebook->menu), "clicked_tab", gtkconv);
+		return FALSE;
+	}
+
+	g_object_set_data(G_OBJECT(notebook->menu), "clicked_tab", gtkconv);
+
+	menu = notebook->menu;
+	gaim_separator(GTK_WIDGET(menu));
+
+	item = gtk_menu_item_new_with_label(_("Close other tabs"));
+	gtk_widget_show(item);
+	gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+	g_signal_connect(G_OBJECT(item), "activate",
+					G_CALLBACK(close_others_cb), menu);
+
+	item = gtk_menu_item_new_with_label(_("Close all tabs"));
+	gtk_widget_show(item);
+	gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+	g_signal_connect(G_OBJECT(item), "activate",
+					G_CALLBACK(close_window), win);
+
+	gaim_separator(menu);
+
+	item = gtk_menu_item_new_with_label(_("Detach this tab"));
+	gtk_widget_show(item);
+	gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+	g_signal_connect(G_OBJECT(item), "activate",
+					G_CALLBACK(detach_tab_cb), menu);
+
+	item = gtk_menu_item_new_with_label(_("Close this tab"));
+	gtk_widget_show(item);
+	gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+	g_signal_connect(G_OBJECT(item), "activate",
+					G_CALLBACK(close_tab_cb), menu);
+
+	return FALSE;
+}
+
+static void
+switch_conv_cb(GtkNotebook *notebook, GtkWidget *page, gint page_num,
+               gpointer user_data)
+{
+	GaimGtkWindow *win;
+	GaimConversation *conv;
+	GaimGtkConversation *gtkconv;
+	const char *sound_method;
+
+	win = user_data;
+	gtkconv = gaim_gtk_conv_window_get_gtkconv_at_index(win, page_num);
+	conv = gtkconv->active_conv;
+
+	g_return_if_fail(conv != NULL);
+
+	/* clear unseen flag if conversation is not hidden */
+	if(!gaim_gtkconv_is_hidden(gtkconv)) {
+		gtkconv_set_unseen(gtkconv, GAIM_UNSEEN_NONE);
+	}
+
+	/* Update the menubar */
+
+	gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(gtkconv->win->menu.logging),
+	                               gaim_conversation_is_logging(conv));
+
+	generate_send_to_items(win);
+	regenerate_options_items(win);
+
+	gaim_gtkconv_switch_active_conversation(conv);
+
+	sound_method = gaim_prefs_get_string("/gaim/gtk/sound/method");
+	if (strcmp(sound_method, "none") != 0)
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.sounds),
+		                               gtkconv->make_sound);
+
+	gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_formatting_toolbar),
+	                               gaim_prefs_get_bool("/gaim/gtk/conversations/show_formatting_toolbar"));
+
+	gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_timestamps),
+	                               gaim_prefs_get_bool("/gaim/gtk/conversations/show_timestamps"));
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM &&
+	    gaim_prefs_get_bool("/gaim/gtk/conversations/im/show_buddy_icons"))
+	{
+		gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(win->menu.show_icon),
+		                               gtkconv->u.im->show_icon);
+	}
+
+	/*
+	 * We pause icons when they are not visible.  If this icon should
+	 * be animated then start it back up again.
+	 */
+	if ((gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM) &&
+	    (gtkconv->u.im->animate))
+		start_anim(NULL, gtkconv);
+
+	gaim_signal_emit(gaim_gtk_conversations_get_handle(), "conversation-switched", conv);
+}
+
+/**************************************************************************
+ * GTK+ window ops
+ **************************************************************************/
+
+GList *
+gaim_gtk_conv_windows_get_list()
+{
+	return window_list;
+}
+
+GaimGtkWindow *
+gaim_gtk_conv_window_new()
+{
+	GaimGtkWindow *win;
+	GtkPositionType pos;
+	GtkWidget *testidea;
+	GtkWidget *menubar;
+
+	win = g_malloc0(sizeof(GaimGtkWindow));
+
+	window_list = g_list_append(window_list, win);
+
+	/* Create the window. */
+	win->window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
+	gtk_window_set_role(GTK_WINDOW(win->window), "conversation");
+	gtk_window_set_resizable(GTK_WINDOW(win->window), TRUE);
+	gtk_container_set_border_width(GTK_CONTAINER(win->window), 0);
+	GTK_WINDOW(win->window)->allow_shrink = TRUE;
+
+	g_signal_connect(G_OBJECT(win->window), "delete_event",
+	                 G_CALLBACK(close_win_cb), win);
+
+	g_signal_connect(G_OBJECT(win->window), "focus_in_event",
+	                 G_CALLBACK(focus_win_cb), win);
+
+	/* Create the notebook. */
+	win->notebook = gtk_notebook_new();
+
+	pos = gaim_prefs_get_int("/gaim/gtk/conversations/tab_side");
+
+#if 0
+	gtk_notebook_set_tab_hborder(GTK_NOTEBOOK(win->notebook), 0);
+	gtk_notebook_set_tab_vborder(GTK_NOTEBOOK(win->notebook), 0);
+#endif
+	gtk_notebook_set_tab_pos(GTK_NOTEBOOK(win->notebook), pos);
+	gtk_notebook_set_scrollable(GTK_NOTEBOOK(win->notebook), TRUE);
+	gtk_notebook_popup_enable(GTK_NOTEBOOK(win->notebook));
+	gtk_notebook_set_show_tabs(GTK_NOTEBOOK(win->notebook), FALSE);
+	gtk_notebook_set_show_border(GTK_NOTEBOOK(win->notebook), FALSE);
+
+	g_signal_connect(G_OBJECT(win->notebook), "button-press-event",
+					G_CALLBACK(right_click_menu_cb), win);
+
+	gtk_widget_show(win->notebook);
+
+	g_signal_connect(G_OBJECT(win->notebook), "switch_page",
+	                 G_CALLBACK(before_switch_conv_cb), win);
+	g_signal_connect_after(G_OBJECT(win->notebook), "switch_page",
+	                       G_CALLBACK(switch_conv_cb), win);
+
+	/* Setup the tab drag and drop signals. */
+	gtk_widget_add_events(win->notebook,
+	                      GDK_BUTTON1_MOTION_MASK | GDK_LEAVE_NOTIFY_MASK);
+	g_signal_connect(G_OBJECT(win->notebook), "button_press_event",
+	                 G_CALLBACK(notebook_press_cb), win);
+	g_signal_connect(G_OBJECT(win->notebook), "button_release_event",
+	                 G_CALLBACK(notebook_release_cb), win);
+
+	testidea = gtk_vbox_new(FALSE, 0);
+
+	/* Setup the menubar. */
+	menubar = setup_menubar(win);
+	gtk_box_pack_start(GTK_BOX(testidea), menubar, FALSE, TRUE, 0);
+
+	gtk_box_pack_start(GTK_BOX(testidea), win->notebook, TRUE, TRUE, 0);
+
+	gtk_container_add(GTK_CONTAINER(win->window), testidea);
+
+	gtk_widget_show(testidea);
+
+#ifdef _WIN32
+	g_signal_connect(G_OBJECT(win->window), "show",
+	                 G_CALLBACK(gtkwgaim_ensure_onscreen), win->window);
+#endif
+
+	return win;
+}
+
+void
+gaim_gtk_conv_window_destroy(GaimGtkWindow *win)
+{
+	gaim_prefs_disconnect_by_handle(win);
+	window_list = g_list_remove(window_list, win);
+
+	/* Close the "Find" dialog if it's open */
+	if (win->dialogs.search)
+		gtk_widget_destroy(win->dialogs.search);
+
+	gtk_widget_hide_all(win->window);
+
+	if (win->gtkconvs) {
+		while (win->gtkconvs) {
+			GList *nextgtk = win->gtkconvs->next;
+			GaimGtkConversation *gtkconv = win->gtkconvs->data;
+			GList *nextcore = gtkconv->convs->next;
+			GaimConversation *conv = gtkconv->convs->data;
+			gaim_conversation_destroy(conv);
+			if (!nextgtk && !nextcore)
+			/* we'll end up invoking ourselves when we destroy our last child */
+			/* so don't destroy ourselves right now */
+				return;
+		}
+		return;
+	}
+	gtk_widget_destroy(win->window);
+
+	g_object_unref(G_OBJECT(win->menu.item_factory));
+
+	gaim_notify_close_with_handle(win);
+
+	g_free(win);
+}
+
+void
+gaim_gtk_conv_window_show(GaimGtkWindow *win)
+{
+	gtk_widget_show(win->window);
+}
+
+void
+gaim_gtk_conv_window_hide(GaimGtkWindow *win)
+{
+	gtk_widget_hide(win->window);
+}
+
+void
+gaim_gtk_conv_window_raise(GaimGtkWindow *win)
+{
+	gdk_window_raise(GDK_WINDOW(win->window->window));
+}
+
+void
+gaim_gtk_conv_window_switch_gtkconv(GaimGtkWindow *win, GaimGtkConversation *gtkconv)
+{
+	gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook),
+	                              gtk_notebook_page_num(GTK_NOTEBOOK(win->notebook),
+		                              gtkconv->tab_cont));
+}
+
+void
+gaim_gtk_conv_window_add_gtkconv(GaimGtkWindow *win, GaimGtkConversation *gtkconv)
+{
+	GaimConversation *conv = gtkconv->active_conv;
+	GaimGtkConversation *focus_gtkconv;
+	GtkWidget *tabby, *menu_tabby;
+	GtkWidget *tab_cont = gtkconv->tab_cont;
+	GtkWidget *close_image;
+	GaimConversationType conv_type;
+	const gchar *tmp_lab;
+	gint close_button_width, close_button_height, focus_width, focus_pad;
+	gboolean tabs_side = FALSE;
+	gint angle = 0;
+
+	conv_type = gaim_conversation_get_type(conv);
+
+
+	win->gtkconvs = g_list_append(win->gtkconvs, gtkconv);
+	gtkconv->win = win;
+
+	if (gaim_prefs_get_int("/gaim/gtk/conversations/tab_side") == GTK_POS_LEFT ||
+	    gaim_prefs_get_int("/gaim/gtk/conversations/tab_side") == GTK_POS_RIGHT)
+		tabs_side = TRUE;
+	else if (gaim_prefs_get_int("/gaim/gtk/conversations/tab_side") == (GTK_POS_LEFT|8))
+		angle = 90;
+	else if (gaim_prefs_get_int("/gaim/gtk/conversations/tab_side") == (GTK_POS_RIGHT|8))
+		angle = 270;
+
+	if (angle)
+		gtkconv->tabby = tabby = gtk_vbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	else
+		gtkconv->tabby = tabby = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+	gtkconv->menu_tabby = menu_tabby = gtk_hbox_new(FALSE, GAIM_HIG_BOX_SPACE);
+
+	/* Close button. */
+	gtkconv->close = gtk_button_new();
+	gtk_icon_size_lookup(GTK_ICON_SIZE_MENU, &close_button_width, &close_button_height);
+	if (gtk_check_version(2, 4, 2) == NULL) {
+		/* Need to account for extra padding around the gtkbutton */
+		gtk_widget_style_get(GTK_WIDGET(gtkconv->close),
+		                     "focus-line-width", &focus_width,
+		                     "focus-padding", &focus_pad,
+		                     NULL);
+		close_button_width += (focus_width + focus_pad) * 2;
+		close_button_height += (focus_width + focus_pad) * 2;
+	}
+	gtk_widget_set_size_request(GTK_WIDGET(gtkconv->close),
+	                            close_button_width, close_button_height);
+
+	gtk_button_set_relief(GTK_BUTTON(gtkconv->close), GTK_RELIEF_NONE);
+	close_image = gtk_image_new_from_stock(GTK_STOCK_CLOSE, GTK_ICON_SIZE_MENU);
+	gtk_widget_show(close_image);
+	gtk_container_add(GTK_CONTAINER(gtkconv->close), close_image);
+	gtk_tooltips_set_tip(gtkconv->tooltips, gtkconv->close,
+	                     _("Close conversation"), NULL);
+
+	g_signal_connect(G_OBJECT(gtkconv->close), "clicked",
+	                 G_CALLBACK(close_conv_cb), gtkconv);
+
+#if !GTK_CHECK_VERSION(2,6,0)
+	/*
+	* I love Galeon. They have a fix for that stupid annoying visible
+	* border bug. I love you guys! -- ChipX86
+	*/
+	/* This is fixed properly in some version of Gtk before 2.6.0  */
+	g_signal_connect(G_OBJECT(gtkconv->close), "state_changed",
+	                 G_CALLBACK(tab_close_button_state_changed_cb), NULL);
+#endif
+
+	/* Status icon. */
+	gtkconv->icon = gtk_image_new();
+	gtkconv->menu_icon = gtk_image_new();
+	update_tab_icon(conv);
+
+	/* Tab label. */
+	gtkconv->tab_label = gtk_label_new(tmp_lab = gaim_conversation_get_title(conv));
+
+#if GTK_CHECK_VERSION(2,6,0)
+	if (!angle)
+		g_object_set(G_OBJECT(gtkconv->tab_label), "ellipsize", PANGO_ELLIPSIZE_END, NULL);
+	gtk_label_set_width_chars(GTK_LABEL(gtkconv->tab_label), 6);
+	if (tabs_side) {
+		gtk_label_set_width_chars(GTK_LABEL(gtkconv->tab_label), MIN(g_utf8_strlen(tmp_lab, -1), 12));
+	}
+	if (angle)
+		gtk_label_set_angle(GTK_LABEL(gtkconv->tab_label), angle);
+#endif
+	gtkconv->menu_label = gtk_label_new(gaim_conversation_get_title(conv));
+#if 0
+	gtk_misc_set_alignment(GTK_MISC(gtkconv->tab_label), 0.00, 0.5);
+	gtk_misc_set_padding(GTK_MISC(gtkconv->tab_label), 4, 0);
+#endif
+
+	/* Pack it all together. */
+	if (angle == 90)
+		gtk_box_pack_start(GTK_BOX(tabby), gtkconv->close, FALSE, FALSE, 0);
+	else
+		gtk_box_pack_start(GTK_BOX(tabby), gtkconv->icon, FALSE, FALSE, 0);
+	gtk_box_pack_start(GTK_BOX(menu_tabby), gtkconv->menu_icon,
+	                   FALSE, FALSE, 0);
+
+	gtk_widget_show_all(gtkconv->icon);
+	gtk_widget_show_all(gtkconv->menu_icon);
+
+	gtk_box_pack_start(GTK_BOX(tabby), gtkconv->tab_label, TRUE, TRUE, 0);
+	gtk_box_pack_start(GTK_BOX(menu_tabby), gtkconv->menu_label, TRUE, TRUE, 0);
+	gtk_widget_show(gtkconv->tab_label);
+	gtk_widget_show(gtkconv->menu_label);
+	gtk_misc_set_alignment(GTK_MISC(gtkconv->menu_label), 0, 0);
+
+	if (angle == 90)
+		gtk_box_pack_start(GTK_BOX(tabby), gtkconv->icon, FALSE, FALSE, 0);
+	else
+		gtk_box_pack_start(GTK_BOX(tabby), gtkconv->close, FALSE, FALSE, 0);
+	if (gaim_prefs_get_bool("/gaim/gtk/conversations/close_on_tabs"))
+		gtk_widget_show(gtkconv->close);
+
+	gtk_widget_show(tabby);
+	gtk_widget_show(menu_tabby);
+
+	if (gaim_conversation_get_type(conv) == GAIM_CONV_TYPE_IM)
+		gaim_gtkconv_update_buddy_icon(conv);
+
+	/* Add this pane to the conversation's notebook. */
+	gtk_notebook_append_page_menu(GTK_NOTEBOOK(win->notebook), tab_cont, tabby, menu_tabby);
+	gtk_notebook_set_tab_label_packing(GTK_NOTEBOOK(win->notebook), tab_cont, !tabs_side && !angle, TRUE, GTK_PACK_START);
+
+
+	gtk_widget_show(tab_cont);
+
+	if (gaim_gtk_conv_window_get_gtkconv_count(win) == 1) {
+		/* Er, bug in notebooks? Switch to the page manually. */
+		gtk_notebook_set_current_page(GTK_NOTEBOOK(win->notebook), 0);
+
+		gtk_notebook_set_show_tabs(GTK_NOTEBOOK(win->notebook),
+		                           gaim_prefs_get_bool("/gaim/gtk/conversations/tabs"));
+	} else
+		gtk_notebook_set_show_tabs(GTK_NOTEBOOK(win->notebook), TRUE);
+
+	focus_gtkconv = g_list_nth_data(gaim_gtk_conv_window_get_gtkconvs(win),
+	                             gtk_notebook_get_current_page(GTK_NOTEBOOK(win->notebook)));
+	gtk_widget_grab_focus(focus_gtkconv->entry);
+
+	if (gaim_gtk_conv_window_get_gtkconv_count(win) == 1)
+		update_send_to_selection(win);
+}
+
+void
+gaim_gtk_conv_window_remove_gtkconv(GaimGtkWindow *win, GaimGtkConversation *gtkconv)
+{
+	unsigned int index;
+	GaimConversationType conv_type;
+
+	conv_type = gaim_conversation_get_type(gtkconv->active_conv);
+	index = gtk_notebook_page_num(GTK_NOTEBOOK(win->notebook), gtkconv->tab_cont);
+
+	g_object_ref(gtkconv->tab_cont);
+	gtk_object_sink(GTK_OBJECT(gtkconv->tab_cont));
+
+	gtk_notebook_remove_page(GTK_NOTEBOOK(win->notebook), index);
+
+	/* go back to tabless if need be */
+	if (gaim_gtk_conv_window_get_gtkconv_count(win) <= 2) {
+		gtk_notebook_set_show_tabs(GTK_NOTEBOOK(win->notebook),
+		                           gaim_prefs_get_bool("/gaim/gtk/conversations/tabs"));
+	}
+
+	win->gtkconvs = g_list_remove(win->gtkconvs, gtkconv);
+
+	if (!win->gtkconvs && win != hidden_convwin)
+		gaim_gtk_conv_window_destroy(win);
+}
+
+GaimGtkConversation *
+gaim_gtk_conv_window_get_gtkconv_at_index(const GaimGtkWindow *win, int index)
+{
+	GtkWidget *tab_cont;
+
+	if (index == -1)
+		index = 0;
+	tab_cont = gtk_notebook_get_nth_page(GTK_NOTEBOOK(win->notebook), index);
+	return tab_cont ? g_object_get_data(G_OBJECT(tab_cont), "GaimGtkConversation") : NULL;
+}
+
+GaimGtkConversation *
+gaim_gtk_conv_window_get_active_gtkconv(const GaimGtkWindow *win)
+{
+	int index;
+	GtkWidget *tab_cont;
+
+	index = gtk_notebook_get_current_page(GTK_NOTEBOOK(win->notebook));
+	if (index == -1)
+		index = 0;
+	tab_cont = gtk_notebook_get_nth_page(GTK_NOTEBOOK(win->notebook), index);
+	if (!tab_cont)
+		return NULL;
+	return g_object_get_data(G_OBJECT(tab_cont), "GaimGtkConversation");
+}
+
+
+GaimConversation *
+gaim_gtk_conv_window_get_active_conversation(const GaimGtkWindow *win)
+{
+	GaimGtkConversation *gtkconv;
+
+	gtkconv = gaim_gtk_conv_window_get_active_gtkconv(win);
+	return gtkconv ? gtkconv->active_conv : NULL;
+}
+
+gboolean
+gaim_gtk_conv_window_is_active_conversation(const GaimConversation *conv)
+{
+	return conv == gaim_gtk_conv_window_get_active_conversation(GAIM_GTK_CONVERSATION(conv)->win);
+}
+
+gboolean
+gaim_gtk_conv_window_has_focus(GaimGtkWindow *win)
+{
+	gboolean has_focus = FALSE;
+
+	g_object_get(G_OBJECT(win->window), "has-toplevel-focus", &has_focus, NULL);
+
+	return has_focus;
+}
+
+GaimGtkWindow *
+gaim_gtk_conv_window_get_at_xy(int x, int y)
+{
+	GaimGtkWindow *win;
+	GdkWindow *gdkwin;
+	GList *l;
+
+	gdkwin = gdk_window_at_pointer(&x, &y);
+
+	if (gdkwin)
+		gdkwin = gdk_window_get_toplevel(gdkwin);
+
+	for (l = gaim_gtk_conv_windows_get_list(); l != NULL; l = l->next) {
+		win = l->data;
+
+		if (gdkwin == win->window->window)
+			return win;
+	}
+
+	return NULL;
+}
+
+GList *
+gaim_gtk_conv_window_get_gtkconvs(GaimGtkWindow *win)
+{
+	return win->gtkconvs;
+}
+
+guint
+gaim_gtk_conv_window_get_gtkconv_count(GaimGtkWindow *win)
+{
+	return g_list_length(win->gtkconvs);
+}
+
+GaimGtkWindow *
+gaim_gtk_conv_window_first_with_type(GaimConversationType type)
+{
+	GList *wins, *convs;
+	GaimGtkWindow *win;
+	GaimGtkConversation *conv;
+
+	if (type == GAIM_CONV_TYPE_UNKNOWN)
+		return NULL;
+
+	for (wins = gaim_gtk_conv_windows_get_list(); wins != NULL; wins = wins->next) {
+		win = wins->data;
+
+		for (convs = win->gtkconvs;
+		     convs != NULL;
+		     convs = convs->next) {
+
+			conv = convs->data;
+
+			if (gaim_conversation_get_type(conv->active_conv) == type)
+				return win;
+		}
+	}
+
+	return NULL;
+}
+
+GaimGtkWindow *
+gaim_gtk_conv_window_last_with_type(GaimConversationType type)
+{
+	GList *wins, *convs;
+	GaimGtkWindow *win;
+	GaimGtkConversation *conv;
+
+	if (type == GAIM_CONV_TYPE_UNKNOWN)
+		return NULL;
+
+	for (wins = g_list_last(gaim_gtk_conv_windows_get_list());
+	     wins != NULL;
+	     wins = wins->prev) {
+
+		win = wins->data;
+
+		for (convs = win->gtkconvs;
+		     convs != NULL;
+		     convs = convs->next) {
+
+			conv = convs->data;
+
+			if (gaim_conversation_get_type(conv->active_conv) == type)
+				return win;
+		}
+	}
+
+	return NULL;
+}
+
+
+/**************************************************************************
+ * Conversation placement functions
+ **************************************************************************/
+typedef struct
+{
+	char *id;
+	char *name;
+	GaimConvPlacementFunc fnc;
+
+} ConvPlacementData;
+
+static GList *conv_placement_fncs = NULL;
+static GaimConvPlacementFunc place_conv = NULL;
+
+/* This one places conversations in the last made window. */
+static void
+conv_placement_last_created_win(GaimGtkConversation *conv)
+{
+	GaimGtkWindow *win;
+
+	GList *l = g_list_last(gaim_gtk_conv_windows_get_list());
+	win = l ? l->data : NULL;;
+
+	if (win == NULL) {
+		win = gaim_gtk_conv_window_new();
+
+		gaim_gtk_conv_window_add_gtkconv(win, conv);
+		gaim_gtk_conv_window_show(win);
+	} else {
+		gaim_gtk_conv_window_add_gtkconv(win, conv);
+	}
+}
+
+/* This one places conversations in the last made window of the same type. */
+static void
+conv_placement_last_created_win_type(GaimGtkConversation *conv)
+{
+	GaimGtkWindow *win;
+
+	win = gaim_gtk_conv_window_last_with_type(gaim_conversation_get_type(conv->active_conv));
+
+	if (win == NULL) {
+		win = gaim_gtk_conv_window_new();
+
+		gaim_gtk_conv_window_add_gtkconv(win, conv);
+		gaim_gtk_conv_window_show(win);
+	} else
+		gaim_gtk_conv_window_add_gtkconv(win, conv);
+}
+
+/* This one places each conversation in its own window. */
+static void
+conv_placement_new_window(GaimGtkConversation *conv)
+{
+	GaimGtkWindow *win;
+
+	win = gaim_gtk_conv_window_new();
+
+	gaim_gtk_conv_window_add_gtkconv(win, conv);
+
+	gaim_gtk_conv_window_show(win);
+}
+
+static GaimGroup *
+conv_get_group(GaimGtkConversation *conv)
+{
+	GaimGroup *group = NULL;
+
+	if (gaim_conversation_get_type(conv->active_conv) == GAIM_CONV_TYPE_IM) {
+		GaimBuddy *buddy;
+
+		buddy = gaim_find_buddy(gaim_conversation_get_account(conv->active_conv),
+		                        gaim_conversation_get_name(conv->active_conv));
+
+		if (buddy != NULL)
+			group = gaim_buddy_get_group(buddy);
+
+	} else if (gaim_conversation_get_type(conv->active_conv) == GAIM_CONV_TYPE_CHAT) {
+		GaimChat *chat;
+
+		chat = gaim_blist_find_chat(gaim_conversation_get_account(conv->active_conv),
+		                            gaim_conversation_get_name(conv->active_conv));
+
+		if (chat != NULL)
+			group = gaim_chat_get_group(chat);
+	}
+
+	return group;
+}
+
+/*
+ * This groups things by, well, group. Buddies from groups will always be
+ * grouped together, and a buddy from a group not belonging to any currently
+ * open windows will get a new window.
+ */
+static void
+conv_placement_by_group(GaimGtkConversation *conv)
+{
+	GaimConversationType type;
+	GaimGroup *group = NULL;
+	GList *wl, *cl;
+
+	type = gaim_conversation_get_type(conv->active_conv);
+
+	group = conv_get_group(conv);
+
+	/* Go through the list of IMs and find one with this group. */
+	for (wl = gaim_gtk_conv_windows_get_list(); wl != NULL; wl = wl->next) {
+		GaimGtkWindow *win2;
+		GaimGtkConversation *conv2;
+		GaimGroup *group2 = NULL;
+
+		win2 = wl->data;
+
+		for (cl = win2->gtkconvs;
+		     cl != NULL;
+		     cl = cl->next) {
+			conv2 = cl->data;
+
+			group2 = conv_get_group(conv2);
+
+			if (group == group2) {
+				gaim_gtk_conv_window_add_gtkconv(win2, conv);
+
+				return;
+			}
+		}
+	}
+
+	/* Make a new window. */
+	conv_placement_new_window(conv);
+}
+
+/* This groups things by account.  Otherwise, the same semantics as above */
+static void
+conv_placement_by_account(GaimGtkConversation *conv)
+{
+	GaimConversationType type;
+	GList *wins, *convs;
+	GaimAccount *account;
+
+	account = gaim_conversation_get_account(conv->active_conv);
+	type = gaim_conversation_get_type(conv->active_conv);
+
+	/* Go through the list of IMs and find one with this group. */
+	for (wins = gaim_gtk_conv_windows_get_list(); wins != NULL; wins = wins->next) {
+		GaimGtkWindow *win2;
+		GaimGtkConversation *conv2;
+
+		win2 = wins->data;
+
+		for (convs = win2->gtkconvs;
+		     convs != NULL;
+		     convs = convs->next) {
+			conv2 = convs->data;
+
+			if (account == gaim_conversation_get_account(conv2->active_conv)) {
+				gaim_gtk_conv_window_add_gtkconv(win2, conv);
+				return;
+			}
+		}
+	}
+
+	/* Make a new window. */
+	conv_placement_new_window(conv);
+}
+
+static ConvPlacementData *
+get_conv_placement_data(const char *id)
+{
+	ConvPlacementData *data = NULL;
+	GList *n;
+
+	for (n = conv_placement_fncs; n; n = n->next) {
+		data = n->data;
+		if (!strcmp(data->id, id))
+			return data;
+	}
+
+	return NULL;
+}
+
+static void
+add_conv_placement_fnc(const char *id, const char *name,
+                       GaimConvPlacementFunc fnc)
+{
+	ConvPlacementData *data;
+
+	data = g_new(ConvPlacementData, 1);
+
+	data->id = g_strdup(id);
+	data->name = g_strdup(name);
+	data->fnc  = fnc;
+
+	conv_placement_fncs = g_list_append(conv_placement_fncs, data);
+}
+
+static void
+ensure_default_funcs(void)
+{
+	if (conv_placement_fncs == NULL) {
+		add_conv_placement_fnc("last", _("Last created window"),
+		                       conv_placement_last_created_win);
+		add_conv_placement_fnc("im_chat", _("Separate IM and Chat windows"),
+		                       conv_placement_last_created_win_type);
+		add_conv_placement_fnc("new", _("New window"),
+		                       conv_placement_new_window);
+		add_conv_placement_fnc("group", _("By group"),
+		                       conv_placement_by_group);
+		add_conv_placement_fnc("account", _("By account"),
+		                       conv_placement_by_account);
+	}
+}
+
+GList *
+gaim_gtkconv_placement_get_options(void)
+{
+	GList *n, *list = NULL;
+	ConvPlacementData *data;
+
+	ensure_default_funcs();
+
+	for (n = conv_placement_fncs; n; n = n->next) {
+		data = n->data;
+		list = g_list_append(list, data->name);
+		list = g_list_append(list, data->id);
+	}
+
+	return list;
+}
+
+
+void
+gaim_gtkconv_placement_add_fnc(const char *id, const char *name,
+                            GaimConvPlacementFunc fnc)
+{
+	g_return_if_fail(id   != NULL);
+	g_return_if_fail(name != NULL);
+	g_return_if_fail(fnc  != NULL);
+
+	ensure_default_funcs();
+
+	add_conv_placement_fnc(id, name, fnc);
+}
+
+void
+gaim_gtkconv_placement_remove_fnc(const char *id)
+{
+	ConvPlacementData *data = get_conv_placement_data(id);
+
+	if (data == NULL)
+		return;
+
+	conv_placement_fncs = g_list_remove(conv_placement_fncs, data);
+
+	g_free(data->id);
+	g_free(data->name);
+	g_free(data);
+}
+
+const char *
+gaim_gtkconv_placement_get_name(const char *id)
+{
+	ConvPlacementData *data;
+
+	ensure_default_funcs();
+
+	data = get_conv_placement_data(id);
+
+	if (data == NULL)
+		return NULL;
+
+	return data->name;
+}
+
+GaimConvPlacementFunc
+gaim_gtkconv_placement_get_fnc(const char *id)
+{
+	ConvPlacementData *data;
+
+	ensure_default_funcs();
+
+	data = get_conv_placement_data(id);
+
+	if (data == NULL)
+		return NULL;
+
+	return data->fnc;
+}
+
+void
+gaim_gtkconv_placement_set_current_func(GaimConvPlacementFunc func)
+{
+	g_return_if_fail(func != NULL);
+
+	/* If tabs are enabled, set the function, otherwise, NULL it out. */
+	if (gaim_prefs_get_bool("/gaim/gtk/conversations/tabs"))
+		place_conv = func;
+	else
+		place_conv = NULL;
+}
+
+GaimConvPlacementFunc
+gaim_gtkconv_placement_get_current_func(void)
+{
+	return place_conv;
+}
+
+void
+gaim_gtkconv_placement_place(GaimGtkConversation *gtkconv)
+{
+	if (place_conv)
+		place_conv(gtkconv);
+	else
+		conv_placement_new_window(gtkconv);
+}
+
+gboolean
+gaim_gtkconv_is_hidden(GaimGtkConversation *gtkconv)
+{
+	g_return_val_if_fail(gtkconv != NULL, FALSE);
+
+	return (gtkconv->win == hidden_convwin);
+}
+
+
+/* Algorithm from http://www.w3.org/TR/AERT#color-contrast */
+static gboolean
+color_is_visible(GdkColor foreground, GdkColor background, int color_contrast, int brightness_contrast)
+{
+	gulong fg_brightness;
+	gulong bg_brightness;
+	gulong br_diff;
+	gulong col_diff;
+	int fred, fgreen, fblue, bred, bgreen, bblue;
+
+	/* this algorithm expects colors between 0 and 255 for each of red green and blue.
+	 * GTK on the other hand has values between 0 and 65535
+	 * Err suggested I >> 8, which grabbed the high bits.
+	 */
+
+	fred = foreground.red >> 8 ;
+	fgreen = foreground.green >> 8 ;
+	fblue = foreground.blue >> 8 ;
+
+
+	bred = background.red >> 8 ;
+	bgreen = background.green >> 8 ;
+	bblue = background.blue >> 8 ;
+
+	fg_brightness = (fred * 299 + fgreen * 587 + fblue * 114) / 1000;
+	bg_brightness = (bred * 299 + bgreen * 587 + bblue * 114) / 1000;
+	br_diff = abs(fg_brightness - bg_brightness);
+
+	col_diff = abs(fred - bred) + abs(fgreen - bgreen) + abs(fblue - bblue);
+
+	return ((col_diff > color_contrast) && (br_diff > brightness_contrast));
+}
+
+
+static GdkColor*
+generate_nick_colors(guint *color_count, GdkColor background)
+{
+	guint numcolors = *color_count;
+	guint i = 0, j = 0;
+	GdkColor *colors = g_new(GdkColor, numcolors);
+	GdkColor nick_highlight;
+	GdkColor send_color;
+	time_t breakout_time;
+
+	gdk_color_parse(HIGHLIGHT_COLOR, &nick_highlight);
+	gdk_color_parse(SEND_COLOR, &send_color);
+
+	srand(background.red + background.green + background.blue + 1);
+
+	breakout_time = time(NULL) + 3;
+
+	/* first we look through the list of "good" colors: colors that differ from every other color in the
+	 * list.  only some of them will differ from the background color though. lets see if we can find
+	 * numcolors of them that do
+	 */
+	while (i < numcolors && j < NUM_NICK_SEED_COLORS && time(NULL) < breakout_time)
+	{
+		GdkColor color = nick_seed_colors[j];
+
+		if (color_is_visible(color, background,     MIN_COLOR_CONTRAST,     MIN_BRIGHTNESS_CONTRAST) &&
+			color_is_visible(color, nick_highlight, MIN_COLOR_CONTRAST / 2, 0) &&
+			color_is_visible(color, send_color,     MIN_COLOR_CONTRAST / 4, 0))
+		{
+			colors[i] = color;
+			i++;
+		}
+		j++;
+	}
+
+	/* we might not have found numcolors in the last loop.  if we did, we'll never enter this one.
+	 * if we did not, lets just find some colors that don't conflict with the background.  its
+	 * expensive to find colors that not only don't conflict with the background, but also do not
+	 * conflict with each other.
+	 */
+	while(i < numcolors && time(NULL) < breakout_time)
+	{
+		GdkColor color = { 0, rand() % 65536, rand() % 65536, rand() % 65536 };
+
+		if (color_is_visible(color, background,     MIN_COLOR_CONTRAST,     MIN_BRIGHTNESS_CONTRAST) &&
+			color_is_visible(color, nick_highlight, MIN_COLOR_CONTRAST / 2, 0) &&
+			color_is_visible(color, send_color,     MIN_COLOR_CONTRAST / 4, 0))
+		{
+			colors[i] = color;
+			i++;
+		}
+	}
+
+	if (i < numcolors) {
+		GdkColor *c = colors;
+		gaim_debug_warning("gtkconv", "Unable to generate enough random colors before timeout. %u colors found.\n", i);
+		colors = g_memdup(c, i * sizeof(GdkColor));
+		g_free(c);
+		*color_count = i;
+	}
+
+	return colors;
+}