Mercurial > pidgin.yaz
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)