changeset 32525:d0be198c4694

other files that missed my main major commit.
author tdrhq@soc.pidgin.im
date Fri, 07 Aug 2009 18:29:52 +0000
parents b3bc3cadc679
children a2fd5f08ed49
files pidgin/plugins/adiumthemes/Makefile.am pidgin/plugins/adiumthemes/Template.html pidgin/plugins/adiumthemes/webkit.c
diffstat 3 files changed, 930 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pidgin/plugins/adiumthemes/Makefile.am	Fri Aug 07 18:29:52 2009 +0000
@@ -0,0 +1,28 @@
+
+webkittemplatedir = $(datadir)/pidgin/webkit
+webkittemplate_DATA = Template.html
+
+webkitdir = $(libdir)/pidgin
+
+webkit_la_LDFLAGS = -module -avoid-version
+
+EXTRA_DIST = $(webkittemplate_DATA)
+
+if PLUGINS
+
+webkit_LTLIBRARIES = webkit.la
+
+webkit_la_SOURCES = webkit.c
+
+endif
+
+webkit_la_LIBADD = $(GTK_LIBS) $(WEBKIT_LIBS)
+
+AM_CPPFLAGS = \
+	-DDATADIR=\"$(datadir)\" \
+	-I$(top_srcdir)/libpurple \
+	-I$(top_builddir)/libpurple \
+	-I$(top_srcdir)/pidgin \
+	$(DEBUG_CFLAGS) \
+	$(GTK_CFLAGS) \
+	$(WEBKIT_CFLAGS)
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pidgin/plugins/adiumthemes/Template.html	Fri Aug 07 18:29:52 2009 +0000
@@ -0,0 +1,164 @@
+<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
+<html>
+<head>
+	<meta http-equiv="content-type" content="text/html; charset=utf-8" />
+	<base href="%@">
+	<script type="text/ecmascript" defer="defer">
+	
+		//Appending new content to the message view
+		function appendMessage(html) {
+			shouldScroll = nearBottom();
+		
+			//Remove any existing insertion point
+			insert = document.getElementById("insert");
+			if(insert) insert.parentNode.removeChild(insert);
+
+			//Append the new message to the bottom of our chat block
+			chat = document.getElementById("Chat");
+			range = document.createRange();
+			range.selectNode(chat);
+			documentFragment = range.createContextualFragment(html);
+			chat.appendChild(documentFragment);
+			
+			alignChat(shouldScroll);
+		}
+		function appendMessageNoScroll(html) {
+			//Remove any existing insertion point
+			insert = document.getElementById("insert");
+			if(insert) insert.parentNode.removeChild(insert);
+
+			//Append the new message to the bottom of our chat block
+			chat = document.getElementById("Chat");
+			range = document.createRange();
+			range.selectNode(chat);
+			documentFragment = range.createContextualFragment(html);
+			chat.appendChild(documentFragment);
+		}
+		function appendNextMessage(html){
+			shouldScroll = nearBottom();
+
+			//Locate the insertion point
+			insert = document.getElementById("insert");
+		
+			//make new node
+			range = document.createRange();
+			range.selectNode(insert.parentNode);
+			newNode = range.createContextualFragment(html);
+
+			//swap
+			insert.parentNode.replaceChild(newNode,insert);
+			
+			alignChat(shouldScroll);
+		}
+		function appendNextMessageNoScroll(html){
+			//Locate the insertion point
+			insert = document.getElementById("insert");
+		
+			//make new node
+			range = document.createRange();
+			range.selectNode(insert.parentNode);
+			newNode = range.createContextualFragment(html);
+
+			//swap
+			insert.parentNode.replaceChild(newNode,insert);
+		}
+		
+		//Auto-scroll to bottom.  Use nearBottom to determine if a scrollToBottom is desired.
+		function nearBottom() {
+			return ( document.body.scrollTop >= ( document.body.offsetHeight - ( window.innerHeight * 1.2 ) ) );
+		}
+		function scrollToBottom() {
+			document.body.scrollTop = document.body.offsetHeight;
+		}
+
+		//Dynamically exchange the active stylesheet
+		function setStylesheet( id, url ) {
+			code = "<style id=\"" + id + "\" type=\"text/css\" media=\"screen,print\">";
+			if( url.length ) code += "@import url( \"" + url + "\" );";
+			code += "</style>";
+			range = document.createRange();
+			head = document.getElementsByTagName( "head" ).item(0);
+			range.selectNode( head );
+			documentFragment = range.createContextualFragment( code );
+			head.removeChild( document.getElementById( id ) );
+			head.appendChild( documentFragment );
+		}
+		
+		//Swap an image with its alt-tag text on click, or expand/unexpand an attached image
+		document.onclick = imageCheck;
+		function imageCheck() {		
+			node = event.target;
+			if(node.tagName == 'IMG' && !client.zoomImage(node) && node.alt) {
+				a = document.createElement('a');
+				a.setAttribute('onclick', 'imageSwap(this)');
+				a.setAttribute('src', node.getAttribute('src'));
+				a.className = node.className;
+				text = document.createTextNode(node.alt);
+				a.appendChild(text);
+				node.parentNode.replaceChild(a, node);
+			}
+		}
+
+		function imageSwap(node) {
+			shouldScroll = nearBottom();
+
+			//Swap the image/text
+			img = document.createElement('img');
+			img.setAttribute('src', node.getAttribute('src'));
+			img.setAttribute('alt', node.firstChild.nodeValue);
+			img.className = node.className;
+			node.parentNode.replaceChild(img, node);
+			
+			alignChat(shouldScroll);
+		}
+		
+		//Align our chat to the bottom of the window.  If true is passed, view will also be scrolled down
+		function alignChat(shouldScroll) {
+			var windowHeight = window.innerHeight;
+			
+			if (windowHeight > 0) {
+				var contentElement = document.getElementById('Chat');
+				var contentHeight = contentElement.offsetHeight;
+				if (windowHeight - contentHeight > 0) {
+					contentElement.style.position = 'relative';
+					contentElement.style.top = (windowHeight - contentHeight) + 'px';
+				} else {
+					contentElement.style.position = 'static';
+				}
+			}
+			
+			if (shouldScroll) scrollToBottom();
+		}
+		
+		function windowDidResize(){
+			alignChat(true/*nearBottom()*/); //nearBottom buggy with inactive tabs
+		}
+		
+		window.onresize = windowDidResize;
+	</script>
+	
+	<style type="text/css">
+		.actionMessageUserName:before { content:"*"; }
+		.actionMessageBody:after { content:"*"; }
+		*{ word-wrap:break-word; }
+		img.scaledToFitImage { height:auto; width:100%; }
+	</style>
+	
+	<!-- This style is shared by all variants. !-->
+	<style id="baseStyle" type="text/css" media="screen,print">	
+		%@
+	</style>
+	
+	<!-- Although we call this mainStyle for legacy reasons, it's actually the variant style !-->
+	<style id="mainStyle" type="text/css" media="screen,print">	
+		@import url( "%@" );
+	</style>
+
+</head>
+<body onload="alignChat(true);" style="==bodyBackground==">
+%@
+<div id="Chat">
+</div>
+%@
+</body>
+</html>
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pidgin/plugins/adiumthemes/webkit.c	Fri Aug 07 18:29:52 2009 +0000
@@ -0,0 +1,738 @@
+/*
+ * Adium Webkit views
+ * Copyright (C) 2007
+ *
+ * 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.
+ */
+
+/* This plugins is basically Sadrul's x-chat plugin, but with Webkit
+ * instead of xtext.
+ */
+
+#define PLUGIN_ID		"gtk-webview-adium-ims"
+#define PLUGIN_NAME		"webview-adium-ims"
+#define PLUGIN_AUTHOR		"Sean Egan <seanegan@gmail.com>"
+#define PURPLE_PLUGINS          "Hell yeah"
+
+/* System headers */
+#include <string.h>
+#include <gdk/gdk.h>
+#include <gtk/gtk.h>
+
+#include <webkit/webkit.h>
+
+/* Purple headers */
+#include <conversation.h>
+#include <notify.h>
+#include <util.h>
+#include <version.h>
+
+/* Pidgin headers */
+#include <gtkconv.h>
+#include <gtkplugin.h>
+#include <smileyparser.h>
+
+static PurpleConversationUiOps *uiops = NULL;
+
+static void (*default_write_conv)(PurpleConversation *conv, const char *name, const char *alias,
+						   const char *message, PurpleMessageFlags flags, time_t mtime);
+static void (*default_create_conversation)(PurpleConversation *conv);
+
+static void (*default_destroy_conversation)(PurpleConversation *conv);
+
+/* Cache the contents of the HTML files */
+char *template_html = NULL;                 /* This is the skeleton: some basic javascript mostly */
+char *header_html = NULL;                   /* This is the first thing to be appended to any conversation */
+char *footer_html = NULL;		    /* This is the last thing appended to the conversation */
+char *incoming_content_html = NULL;         /* This is a received message */
+char *outgoing_content_html = NULL;         /* And a sent one */
+char *incoming_next_content_html = NULL;    /* The same things, but used when someone sends multiple subsequent */
+char *outgoing_next_content_html = NULL;    /* messages in a row */
+char *status_html = NULL;                   /* Non-IM status messages */
+char *basestyle_css = NULL;		    /* Shared CSS attributes */
+
+/* Cache their lenghts too, to pass into g_string_new_len, avoiding crazy allocation */
+gsize template_html_len = 0;
+gsize header_html_len = 0;
+gsize footer_html_len = 0;
+gsize incoming_content_html_len = 0;
+gsize outgoing_content_html_len = 0;
+gsize incoming_next_content_html_len = 0;
+gsize outgoing_next_content_html_len = 0;
+gsize status_html_len = 0;
+gsize basestyle_css_len = 0;
+
+/* And their paths */
+char *style_dir = NULL;
+char *template_path = NULL;
+char *css_path = NULL;
+
+static char *
+replace_message_tokens(char *text, gsize len, PurpleConversation *conv, const char *name, const char *alias, 
+			     const char *message, PurpleMessageFlags flags, time_t mtime)
+{
+	GString *str = g_string_new_len(NULL, len);
+	char *cur = text;
+	char *prev = cur;
+
+	while ((cur = strchr(cur, '%'))) {
+		const char *replace = NULL;
+		char *fin = NULL;
+			
+		if (!strncmp(cur, "%message%", strlen("%message%"))) {
+			replace = message;
+		} else if (!strncmp(cur, "%messageClasses%", strlen("%messageClasses%"))) {
+			replace = flags & PURPLE_MESSAGE_SEND ? "outgoing" :
+				  flags & PURPLE_MESSAGE_RECV ? "incoming" : "event";
+		} else if (!strncmp(cur, "%time", strlen("%time"))) {
+			char *format = NULL;
+			if (*(cur + strlen("%time")) == '{') {
+				char *start = cur + strlen("%time") + 1;
+				char *end = strstr(start, "}%");
+				if (!end) /* Invalid string */
+					continue;
+				format = g_strndup(start, end - start);
+				fin = end + 1;
+			} 
+			replace = purple_utf8_strftime(format ? format : "%X", NULL);
+			g_free(format);
+		} else if (!strncmp(cur, "%userIconPath%", strlen("%userIconPath%"))) {
+			if (flags & PURPLE_MESSAGE_SEND) {
+				if (purple_account_get_bool(conv->account, "use-global-buddyicon", TRUE)) {
+					replace = purple_prefs_get_path(PIDGIN_PREFS_ROOT "/accounts/buddyicon");
+				} else {
+					PurpleStoredImage *img = purple_buddy_icons_find_account_icon(conv->account);
+					replace = purple_imgstore_get_filename(img);
+				}
+				if (replace == NULL || !g_file_test(replace, G_FILE_TEST_EXISTS)) {
+					replace = g_build_filename("Outgoing", "buddy_icon.png", NULL);
+				}
+			} else if (flags & PURPLE_MESSAGE_RECV) {
+				PurpleBuddyIcon *icon = purple_conv_im_get_icon(PURPLE_CONV_IM(conv));
+				replace = purple_buddy_icon_get_full_path(icon);
+				if (replace == NULL || !g_file_test(replace, G_FILE_TEST_EXISTS)) {
+					replace = g_build_filename("Incoming", "buddy_icon.png", NULL);
+				}
+			}
+			
+		} else if (!strncmp(cur, "%senderScreenName%", strlen("%senderScreenName%"))) {
+			replace = name;
+		} else if (!strncmp(cur, "%sender%", strlen("%sender%"))) {
+			replace = alias;
+		} else if (!strncmp(cur, "%service%", strlen("%service%"))) {
+			replace = purple_account_get_protocol_name(conv->account);
+		} else {
+			cur++;
+			continue;
+		}
+
+		/* Here we have a replacement to make */
+		g_string_append_len(str, prev, cur - prev);
+		g_string_append(str, replace);
+
+		/* And update the pointers */
+		if (fin) {
+			prev = cur = fin + 1;	
+		} else {
+			prev = cur = strchr(cur + 1, '%') + 1;
+		}
+
+	}
+	
+	/* And wrap it up */
+	g_string_append(str, prev);
+	return g_string_free(str, FALSE);
+}
+
+static char *
+replace_header_tokens(char *text, gsize len, PurpleConversation *conv)
+{
+	GString *str = g_string_new_len(NULL, len);
+	char *cur = text;
+	char *prev = cur;
+
+	if (text == NULL)
+		return NULL;
+
+	while ((cur = strchr(cur, '%'))) {
+		const char *replace = NULL;
+		char *fin = NULL;
+
+  		if (!strncmp(cur, "%chatName%", strlen("%chatName%"))) {
+			replace = conv->name;
+  		} else if (!strncmp(cur, "%sourceName%", strlen("%sourceName%"))) {
+			replace = purple_account_get_alias(conv->account);
+			if (replace == NULL)
+				replace = purple_account_get_username(conv->account);
+  		} else if (!strncmp(cur, "%destinationName%", strlen("%destinationName%"))) {
+			PurpleBuddy *buddy = purple_find_buddy(conv->account, conv->name);
+			if (buddy) {
+				replace = purple_buddy_get_alias(buddy);
+			} else {
+				replace = conv->name;
+			}
+  		} else if (!strncmp(cur, "%incomingIconPath%", strlen("%incomingIconPath%"))) {
+			PurpleBuddyIcon *icon = purple_conv_im_get_icon(PURPLE_CONV_IM(conv));
+			replace = purple_buddy_icon_get_full_path(icon);
+  		} else if (!strncmp(cur, "%outgoingIconPath%", strlen("%outgoingIconPath%"))) {
+  		} else if (!strncmp(cur, "%timeOpened", strlen("%timeOpened"))) {
+			char *format = NULL;
+			if (*(cur + strlen("%timeOpened")) == '{') {
+				char *start = cur + strlen("%timeOpened") + 1;
+				char *end = strstr(start, "}%");
+				if (!end) /* Invalid string */
+					continue;
+				format = g_strndup(start, end - start);
+				fin = end + 1;
+			} 
+			replace = purple_utf8_strftime(format ? format : "%X", NULL);
+			g_free(format);
+		} else {
+		//	cur++;
+			continue;
+		}
+
+		/* Here we have a replacement to make */
+		g_string_append_len(str, prev, cur - prev);
+		g_string_append(str, replace);
+
+		/* And update the pointers */
+		if (fin) {
+			prev = cur = fin + 1;	
+		} else {
+			prev = cur = strchr(cur + 1, '%') + 1;
+		}
+	}
+	
+	/* And wrap it up */
+	g_string_append(str, prev);
+	return g_string_free(str, FALSE);
+}
+
+static char *
+replace_template_tokens(char *text, int len, char *header, char *footer) {
+	GString *str = g_string_new_len(NULL, len);
+
+	char **ms = g_strsplit(text, "%@", 6);
+	char *base = NULL;
+
+	if (ms[0] == NULL || ms[1] == NULL || ms[2] == NULL || ms[3] == NULL || ms[4] == NULL || ms[5] == NULL) {
+		g_strfreev(ms);
+		g_string_free(str, TRUE);
+		return NULL;
+	}
+
+	g_string_append(str, ms[0]);
+	g_string_append(str, "file://");
+	base = g_build_filename (style_dir, "Contents", "Resources", "Template.html", NULL);
+	g_string_append(str, base);
+	g_free (base);
+
+	g_string_append(str, ms[1]);
+	if (basestyle_css)
+		g_string_append(str, basestyle_css);
+	g_string_append(str, ms[2]);
+	if (css_path) {
+		g_string_append(str, "file://");
+		g_string_append(str, css_path);
+	}
+
+	g_string_append(str, ms[3]);
+	if (header)
+		g_string_append(str, header);
+	g_string_append(str, ms[4]);
+	if (footer)
+		g_string_append(str, footer);
+	g_string_append(str, ms[5]);
+	
+	g_strfreev(ms);
+	return g_string_free (str, FALSE);
+}
+
+static GtkWidget *
+get_webkit(PurpleConversation *conv)
+{
+	PidginConversation *gtkconv;
+	gtkconv = PIDGIN_CONVERSATION(conv);
+	if (!gtkconv)
+		return NULL;
+	else 
+		return gtkconv->webview;
+}
+static void
+init_theme_for_webkit (PurpleConversation *conv)
+{
+	GtkWidget *webkit = PIDGIN_CONVERSATION(conv)->webview;
+	char *header, *footer;
+	char *template;
+
+	char* basedir = g_build_filename (style_dir, "Contents", "Resources", "Template.html", NULL);
+	char* baseuri = g_strdup_printf ("file://%s", basedir);
+
+	header = replace_header_tokens(header_html, header_html_len, conv);
+	footer = replace_header_tokens(footer_html, footer_html_len, conv);
+	template = replace_template_tokens(template_html, template_html_len + header_html_len, header, footer);
+
+	if (!g_object_get_data (G_OBJECT(webkit), "adium-themed"))
+		webkit_web_view_load_string(WEBKIT_WEB_VIEW(webkit), template, "text/html", "UTF-8", baseuri);
+
+	g_object_set_data (G_OBJECT(webkit), "adium-themed", GINT_TO_POINTER(1));
+	g_free (basedir);
+	g_free (baseuri);
+	g_free (header);
+	g_free (footer);
+	g_free (template);
+}
+
+
+/* restore the non theme version of the conversation window */
+static void
+finalize_theme_for_webkit (PurpleConversation *conv)
+{
+	GtkWidget *webview = PIDGIN_CONVERSATION(conv)->webview;
+	webkit_web_view_load_string(WEBKIT_WEB_VIEW(webview), "", "text/html", "UTF-8", "");
+	g_object_set_data (G_OBJECT(webview), "adium-themed", NULL);
+}
+
+static char *
+escape_message(char *text)
+{
+	GString *str = g_string_new(NULL);
+	char *cur = text;
+
+	while (cur && *cur) {
+		switch (*cur) {
+		case '\\':
+			g_string_append(str, "\\\\");	
+			break;
+		case '\"':
+			g_string_append(str, "\\\"");
+			break;
+		case '\r':
+			g_string_append(str, "<br/>");
+			break;
+		case '\n':
+			break;
+		default:
+			g_string_append_c(str, *cur);
+		}
+		cur++;
+	}
+	return g_string_free(str, FALSE);
+}
+
+struct webkit_script {
+	GtkWidget *webkit;
+	char *script;
+};
+
+static gboolean purple_webkit_execute_script(gpointer _script)
+{
+	struct webkit_script *script = (struct webkit_script*) _script;
+	printf ("%s\n", script->script);
+	webkit_web_view_execute_script(WEBKIT_WEB_VIEW(script->webkit), script->script);
+	g_free(script->script);
+	g_free(script);
+	return FALSE;
+}
+
+static void purple_webkit_write_conv(PurpleConversation *conv, const char *name, const char *alias,
+						   const char *message, PurpleMessageFlags flags, time_t mtime)
+{
+	PurpleConversationType type;
+	GtkWidget *webkit;
+	char *stripped;
+	char *message_html;
+	char *msg;
+	char *escape;
+	char *script;
+	char *func = "appendMessage";
+	char *smileyed;
+	struct webkit_script *wk_script;
+	PurpleMessageFlags old_flags = GPOINTER_TO_INT(purple_conversation_get_data(conv, "webkit-lastflags")); 
+
+	/* Don't call the usual stuff first. */
+	//default_write_conv(conv, name, alias, message, flags, mtime);
+	type = purple_conversation_get_type(conv);
+	if (type != PURPLE_CONV_TYPE_IM)
+	{
+		/* If it's chat, we have nothing to do. */
+		return;
+	}
+	/* So it's an IM. Let's play. */
+
+	webkit = get_webkit(conv);
+	stripped = g_strdup(message); //purple_markup_strip_html(message);
+
+	if (flags & PURPLE_MESSAGE_SEND && old_flags & PURPLE_MESSAGE_SEND) {
+		message_html = outgoing_next_content_html;
+		func = "appendNextMessage";
+	} else if (flags & PURPLE_MESSAGE_SEND) {
+		message_html = outgoing_content_html;
+	} else if (flags & PURPLE_MESSAGE_RECV && old_flags & PURPLE_MESSAGE_RECV) {
+		message_html = incoming_next_content_html;
+		func = "appendNextMessage";
+	} else if (flags & PURPLE_MESSAGE_RECV) {
+		message_html = incoming_content_html;
+	} else {
+		message_html = status_html;
+	}
+	purple_conversation_set_data(conv, "webkit-lastflags", GINT_TO_POINTER(flags));
+
+	smileyed = smiley_parse_markup(stripped, conv->account->protocol_id);
+	msg = replace_message_tokens(message_html, 0, conv, name, alias, smileyed, flags, mtime);
+	escape = escape_message(msg);
+	script = g_strdup_printf("%s(\"%s\")", func, escape);
+
+	wk_script = g_new0(struct webkit_script, 1);
+	wk_script->script = script;
+	wk_script->webkit = webkit;
+
+	g_idle_add(purple_webkit_execute_script, wk_script);
+
+	g_free(smileyed);
+	g_free(msg);
+	g_free(stripped);
+	g_free(escape);
+}
+
+
+static void
+purple_webkit_create_conv(PurpleConversation *conv)
+{
+	default_create_conversation(conv);
+	init_theme_for_webkit(conv);
+}
+
+static void
+purple_webkit_destroy_conv(PurpleConversation *conv)
+{
+	default_destroy_conversation(conv);
+}
+
+static GList*
+get_theme_files() 
+{
+	GList *ret = NULL;
+        GDir *variants;
+	char *globe = g_build_filename(DATADIR, "pidgin", "webkit", "styles", NULL);
+	const char *css_file;
+	char *css;
+
+	if (style_dir != NULL) {
+		char *variant_dir = g_build_filename(style_dir, "Contents", "Resources", "Variants", NULL);
+		variants = g_dir_open(variant_dir, 0, NULL);
+		while ((css_file = g_dir_read_name(variants)) != NULL) {
+			if (!strstr(css_file, ".css")) {
+				continue;
+			}
+			css = g_build_filename(variant_dir, css_file, NULL);
+			ret = g_list_append(ret,css);
+		}
+		g_dir_close(variants);
+		g_free(variant_dir);
+	}
+	g_free(globe);
+
+	ret = g_list_sort (ret, (GCompareFunc)g_strcmp0);
+	return ret;	
+}
+
+static void
+variant_set_default ()
+{
+
+	GList* all;
+	GList* copy;
+	css_path = g_strdup (purple_prefs_get_string ("/plugins/gtk/adiumthemes/csspath"));
+
+	if (g_file_test (css_path, G_FILE_TEST_EXISTS))
+		return;
+	else {
+		g_free (css_path);
+		css_path = NULL;
+	}
+	
+	all = get_theme_files ();
+	copy = all;
+	if (css_path) {
+		g_free (css_path);
+		css_path = NULL;
+	}
+	if (all) {
+		css_path = g_strdup (all->data);
+		purple_prefs_set_string ("/plugins/gtk/adiumthemes/csspath", css_path);
+	}
+	
+	while (all) {
+		g_free (all->data);
+		all = g_list_next(all);
+	}
+	g_list_free (copy);
+}
+
+static gboolean
+plugin_load(PurplePlugin *plugin)
+{
+	char *file;
+
+	if (g_path_is_absolute (purple_user_dir())) 
+		style_dir = g_build_filename(purple_user_dir(), "style", NULL);
+	else {
+		char* cur = g_get_current_dir ();
+		style_dir = g_build_filename (cur, purple_user_dir(), "style", NULL);
+		g_free (cur);
+	}
+
+	variant_set_default ();
+	if (!css_path)
+		return FALSE;
+
+
+	template_path = g_build_filename(style_dir, "Contents", "Resources", "Template.html", NULL);
+	if (!g_file_test(template_path, G_FILE_TEST_EXISTS)) {
+		g_free(template_path);
+		template_path = g_build_filename(DATADIR, "pidgin", "webkit", "Template.html", NULL);
+	}
+	if (!g_file_get_contents(template_path, &template_html, &template_html_len, NULL))
+		return FALSE;
+	
+	file = g_build_filename(style_dir, "Contents", "Resources", "Header.html", NULL);
+	g_file_get_contents(file, &header_html, &header_html_len, NULL);
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Footer.html", NULL);
+	g_file_get_contents(file, &footer_html, &footer_html_len, NULL);
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Incoming", "Content.html", NULL);
+	if (!g_file_get_contents(file, &incoming_content_html, &incoming_content_html_len, NULL))
+		return FALSE;
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Incoming", "NextContent.html", NULL);
+	if (!g_file_get_contents(file, &incoming_next_content_html, &incoming_next_content_html_len, NULL)) {
+		incoming_next_content_html = incoming_content_html;
+		incoming_next_content_html_len = incoming_content_html_len;
+	}
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Outgoing", "Content.html", NULL);
+	if (!g_file_get_contents(file, &outgoing_content_html, &outgoing_content_html_len, NULL)) {
+		outgoing_content_html = incoming_content_html;
+		outgoing_content_html_len = incoming_content_html_len;
+	}
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Outgoing", "NextContent.html", NULL);
+	if (!g_file_get_contents(file, &outgoing_next_content_html, &outgoing_next_content_html_len, NULL)) {
+		outgoing_next_content_html = outgoing_content_html;
+		outgoing_next_content_html_len = outgoing_content_html_len;
+	}
+
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "Status.html", NULL);
+	if (!g_file_get_contents(file, &status_html, &status_html_len, NULL))
+		return FALSE;
+
+	file = g_build_filename(style_dir, "Contents", "Resources", "main.css", NULL);
+	g_file_get_contents(file, &basestyle_css, &basestyle_css_len, NULL);
+
+	uiops = pidgin_conversations_get_conv_ui_ops();
+
+	if (uiops == NULL)
+		return FALSE;
+
+	/* Use the oh-so-useful uiops. Signals? bleh. */
+	default_write_conv = uiops->write_conv;
+	uiops->write_conv = purple_webkit_write_conv;
+
+	default_create_conversation = uiops->create_conversation;
+	uiops->create_conversation = purple_webkit_create_conv;
+
+	default_destroy_conversation = uiops->destroy_conversation;
+	uiops->destroy_conversation = purple_webkit_destroy_conv;
+
+	/* finally update each of the existing conversation windows */
+	{
+		GList* list = purple_get_conversations ();
+		for (;list; list = g_list_next(list))
+			purple_webkit_create_conv (list->data);
+			
+	}
+	return TRUE;
+}
+
+static gboolean
+plugin_unload(PurplePlugin *plugin)
+{
+	GList *list;
+	/* Restore the default ui-ops */
+	uiops->write_conv = default_write_conv;
+	uiops->create_conversation = default_create_conversation;
+	uiops->destroy_conversation = default_destroy_conversation;
+
+	list = purple_get_conversations ();
+	while (list) {
+		finalize_theme_for_webkit(list->data);
+		list = g_list_next(list);
+	}
+	return TRUE;
+}
+
+
+static void
+variant_update_conversation (PurpleConversation *conv)
+{
+	PidginConversation *gtkconv = PIDGIN_CONVERSATION (conv);
+	WebKitWebView *webview = WEBKIT_WEB_VIEW (gtkconv->webview);
+	char* script = g_strdup_printf ("setStylesheet(\"mainStyle\",\"%s\")", css_path);
+	printf ("css_path: %s\n", css_path);
+	webkit_web_view_execute_script (webview, script);
+	g_free (script);
+}
+
+static void
+variant_changed (GtkWidget* combobox, gpointer null)
+{
+	char *name, *name_with_ext;
+	GList *list;
+
+	g_free (css_path);
+	name = gtk_combo_box_get_active_text (GTK_COMBO_BOX(combobox));
+	name_with_ext = g_strdup_printf ("%s.css", name);
+	g_free (name);
+	
+	css_path = g_build_filename (style_dir, "Contents", "Resources", "Variants", name_with_ext, NULL);
+	purple_prefs_set_string ("/plugins/gtk/adiumthemes/csspath", css_path);
+	g_free (name_with_ext);
+
+	/* update each conversation */
+	list = purple_get_conversations ();
+	while (list) {
+		variant_update_conversation (list->data);
+		list = g_list_next(list);
+	}
+}
+
+static GtkWidget *
+get_config_frame(PurplePlugin *plugin) {
+	GList *themes = get_theme_files();
+	GList *theme = themes;
+	char *curdir = NULL;
+	GtkWidget *combobox = gtk_combo_box_new_text();	
+	int def = -1, index = 0;
+
+	while (theme) {
+		char *basename = g_path_get_basename(theme->data);
+		char *dirname = g_path_get_dirname(theme->data);
+		if (!curdir || strcmp(curdir, dirname)) {
+			char *plist, *plist_xml;
+			gsize plist_len;
+			xmlnode *node;
+			g_free(curdir);
+			curdir = strdup(dirname);
+			plist = g_build_filename(curdir, "..", "..", "Info.plist", NULL);
+		        if (!g_file_get_contents(plist, &plist_xml, &plist_len, NULL)) {
+				continue;
+			}
+	                node = xmlnode_from_str(plist_xml, plist_len);
+			if (!node) continue;
+			node = xmlnode_get_child(node, "dict");
+			if (!node) continue;
+			node = xmlnode_get_child(node, "key");
+			while (node && strcmp(xmlnode_get_data(node), "CFBundleName")) {
+				node = xmlnode_get_next_twin(node);
+			}
+			if (!node) continue;
+			node = node->next;
+			while (node && node->type != XMLNODE_TYPE_TAG) {
+				node = node->next;
+			}
+
+		}
+		
+		{
+			char *temp = g_strndup (basename, strlen(basename)-4);
+			gtk_combo_box_append_text (GTK_COMBO_BOX(combobox), temp);
+			g_free (temp);
+		}
+		if (g_str_has_suffix (css_path, basename))
+			def = index;
+		index ++;
+		theme = theme->next;
+	}
+
+	gtk_combo_box_set_active (GTK_COMBO_BOX(combobox), def);
+	g_signal_connect (G_OBJECT(combobox), "changed", G_CALLBACK(variant_changed), NULL);
+
+	return combobox;
+}
+
+PidginPluginUiInfo ui_info =
+{
+        get_config_frame,
+        0, /* page_num (Reserved) */
+
+        /* padding */
+        NULL,
+        NULL,
+        NULL,
+        NULL
+};
+
+
+static PurplePluginInfo info =
+{
+	PURPLE_PLUGIN_MAGIC,		/* Magic				*/
+	PURPLE_MAJOR_VERSION,		/* Purple Major Version	*/
+	PURPLE_MINOR_VERSION,		/* Purple Minor Version	*/
+	PURPLE_PLUGIN_STANDARD,		/* plugin type			*/
+PIDGIN_PLUGIN_TYPE,			/* ui requirement		*/
+	0,							/* flags				*/
+	NULL,						/* dependencies			*/
+	PURPLE_PRIORITY_DEFAULT,	/* priority				*/
+
+	PLUGIN_ID,					/* plugin id			*/
+	NULL,						/* name					*/
+	"0.1",					/* version				*/
+	NULL,						/* summary				*/
+	NULL,						/* description			*/
+	PLUGIN_AUTHOR,				/* author				*/
+	"http://pidgin.im",					/* website				*/
+
+	plugin_load,				/* load					*/
+	plugin_unload,				/* unload				*/
+	NULL,						/* destroy				*/
+
+	&ui_info,						/* ui_info				*/
+	NULL,						/* extra_info			*/
+	NULL,						/* prefs_info			*/
+	NULL,						/* actions				*/
+	NULL,						/* reserved 1			*/
+	NULL,						/* reserved 2			*/
+	NULL,						/* reserved 3			*/
+	NULL						/* reserved 4			*/
+};
+
+static void
+init_plugin(PurplePlugin *plugin) {
+	info.name = "Adium IMs";
+	info.summary = "Adium-like IMs with Pidgin";
+	info.description = "You can chat in Pidgin using Adium's WebKit view.";
+
+	purple_prefs_add_none ("/plugins");
+	purple_prefs_add_none ("/plugins/gtk");
+	purple_prefs_add_none ("/plugins/gtk/adiumthemes");
+	purple_prefs_add_string ("/plugins/gtk/adiumthemes/csspath", "");
+}
+
+PURPLE_INIT_PLUGIN(webkit, init_plugin, info)