Mercurial > pidgin.yaz
changeset 15351:682908b965cf
[gaim-migrate @ 18143]
A really great raw XMPP plugin.
committer: Tailor Script <tailor@pidgin.im>
author | Sean Egan <seanegan@gmail.com> |
---|---|
date | Wed, 17 Jan 2007 08:58:26 +0000 |
parents | 635cc5247696 |
children | 70407d31099b |
files | gtk/plugins/Makefile.am gtk/plugins/xmppconsole.c libgaim/protocols/jabber/jabber.c libgaim/xmlnode.c |
diffstat | 4 files changed, 892 insertions(+), 8 deletions(-) [+] |
line wrap: on
line diff
--- a/gtk/plugins/Makefile.am Wed Jan 17 06:29:23 2007 +0000 +++ b/gtk/plugins/Makefile.am Wed Jan 17 08:58:26 2007 +0000 @@ -37,6 +37,7 @@ spellchk_la_LDFLAGS = -module -avoid-version timestamp_la_LDFLAGS = -module -avoid-version timestamp_format_la_LDFLAGS = -module -avoid-version +xmppconsole_la_LDFLAGS = -module -avoid-version if PLUGINS @@ -51,7 +52,8 @@ relnot.la \ spellchk.la \ timestamp.la \ - timestamp_format.la + timestamp_format.la \ + xmppconsole.la convcolors_la_SOURCES = convcolors.c extplacement_la_SOURCES = extplacement.c @@ -64,6 +66,7 @@ spellchk_la_SOURCES = spellchk.c timestamp_la_SOURCES = timestamp.c timestamp_format_la_SOURCES = timestamp_format.c +xmppconsole_la_SOURCES = xmppconsole.c convcolors_la_LIBADD = $(GTK_LIBS) extplacement_la_LIBADD = $(GTK_LIBS) @@ -76,6 +79,7 @@ spellchk_la_LIBADD = $(GTK_LIBS) timestamp_la_LIBADD = $(GTK_LIBS) timestamp_format_la_LIBADD = $(GTK_LIBS) +xmppconsole_la_LIBADD = $(GTK_LIBS) endif # PLUGINS
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/gtk/plugins/xmppconsole.c Wed Jan 17 08:58:26 2007 +0000 @@ -0,0 +1,864 @@ +/* + * Gaim - XMPP debugging tool + * + * Copyright (C) 2002-2003, Sean Egan + * + * 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 "gtkplugin.h" +#include "version.h" +#include "prpl.h" +#include "xmlnode.h" + +#include "gtkimhtml.h" + +typedef struct { + GaimConnection *gc; + GtkWidget *window; + GtkWidget *hbox; + GtkWidget *dropdown; + GtkWidget *imhtml; + GtkWidget *entry; + GtkWidget *sw; + int count; + GList *accounts; +} XmppConsole; + +XmppConsole *console = NULL; +static void *xmpp_console_handle = NULL; + +#define BRACKET_COLOR "#940f8c" +#define TAG_COLOR "#8b1dab" +#define ATTR_NAME_COLOR "#a02961" +#define ATTR_VALUE_COLOR "#324aa4" +#define XMLNS_COLOR "#2cb12f" + +static char * +xmlnode_to_pretty_str(xmlnode *node, int *len, int depth) +{ + GString *text = g_string_new(""); + xmlnode *c; + char *node_name, *esc, *esc2, *tab = NULL; + gboolean need_end = FALSE, pretty = TRUE; + + g_return_val_if_fail(node != NULL, NULL); + + if(pretty && depth) { + tab = g_strnfill(depth, '\t'); + text = g_string_append(text, tab); + } + + node_name = g_markup_escape_text(node->name, -1); + g_string_append_printf(text, "<font color='" + BRACKET_COLOR "'><</font><font color='" + TAG_COLOR "'><b>%s</b></font>", node_name); + + if (node->xmlns) { + if((!node->parent || + !node->parent->xmlns || + strcmp(node->xmlns, node->parent->xmlns)) && + strcmp(node->xmlns, "jabber:client")) + { + char *xmlns = g_markup_escape_text(node->xmlns, -1); + g_string_append_printf(text, " <font color='" + ATTR_NAME_COLOR "'><b>xmlns</b></font>='<font color='" + XMLNS_COLOR "'><b>%s</b></font>'", xmlns); + g_free(xmlns); + } + } + for(c = node->child; c; c = c->next) + { + if(c->type == XMLNODE_TYPE_ATTRIB) { + esc = g_markup_escape_text(c->name, -1); + esc2 = g_markup_escape_text(c->data, -1); + g_string_append_printf(text, " <font color='" + ATTR_NAME_COLOR "'><b>%s</b></font>='<font color='" + ATTR_VALUE_COLOR "'>%s</font>'", esc, esc2); + g_free(esc); + g_free(esc2); + } else if(c->type == XMLNODE_TYPE_TAG || c->type == XMLNODE_TYPE_DATA) { + if(c->type == XMLNODE_TYPE_DATA) + pretty = FALSE; + need_end = TRUE; + } + } + + if(need_end) { + g_string_append_printf(text, + "<font color='"BRACKET_COLOR"'>></font>%s", pretty ? "<br>" : ""); + + for(c = node->child; c; c = c->next) + { + if(c->type == XMLNODE_TYPE_TAG) { + int esc_len; + esc = xmlnode_to_pretty_str(c, &esc_len, depth+1); + text = g_string_append_len(text, esc, esc_len); + g_free(esc); + } else if(c->type == XMLNODE_TYPE_DATA && c->data_sz > 0) { + esc = g_markup_escape_text(c->data, c->data_sz); + text = g_string_append(text, esc); + g_free(esc); + } + } + + if(tab && pretty) + text = g_string_append(text, tab); + g_string_append_printf(text, "<font color='"BRACKET_COLOR"'><</font>/<font color='" + TAG_COLOR"'><b>%s</b></font><font color='"BRACKET_COLOR + "'>></font><br>", node_name); + } else { + g_string_append_printf(text, "/<font color='"BRACKET_COLOR"'>></font><br>"); + } + + g_free(node_name); + + g_free(tab); + + if(len) + *len = text->len; + + return g_string_free(text, FALSE); +} + +static void +xmlnode_received_cb(GaimConnection *gc, xmlnode **packet, gpointer null) +{ + char *str, *formatted; + + if (!console || console->gc != gc) + return; + str = xmlnode_to_pretty_str(*packet, NULL, 0); + formatted = g_strdup_printf("<body bgcolor='#ffcece'><pre>%s</pre></body>", str); + gtk_imhtml_append_text(GTK_IMHTML(console->imhtml), formatted, 0); + g_free(formatted); + g_free(str); +} + +static void +xmlnode_sent_cb(GaimConnection *gc, char **packet, gpointer null) +{ + char *str; + char *formatted; + xmlnode *node; + + if (!console || console->gc != gc) + return; + node = xmlnode_from_str(*packet, -1); + + if (!node) + return; + + str = xmlnode_to_pretty_str(node, NULL, 0); + formatted = g_strdup_printf("<body bgcolor='#dcecc4'><pre>%s</pre></body>", str); + gtk_imhtml_append_text(GTK_IMHTML(console->imhtml), formatted, 0); + g_free(formatted); + g_free(str); + xmlnode_free(node); +} + +static void message_send_cb(GtkWidget *widget, gpointer p) +{ + GtkTextIter start, end; + GaimPluginProtocolInfo *prpl_info = NULL; + GaimConnection *gc = console->gc; + GtkTextBuffer *buffer; + char *text; + + gc = console->gc; + + prpl_info = GAIM_PLUGIN_PROTOCOL_INFO(gc->prpl); + + buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(console->entry)); + gtk_text_buffer_get_start_iter(buffer, &start); + gtk_text_buffer_get_end_iter(buffer, &end); + + text = gtk_imhtml_get_text(GTK_IMHTML(console->entry), &start, &end); + + if (gc && prpl_info->convo_closed != NULL) + prpl_info->send_raw(gc, text, strlen(text)); + + g_free(text); + gtk_imhtml_clear(GTK_IMHTML(console->entry)); +} + +static void entry_changed_cb(GtkTextBuffer *buffer, void *data) +{ + char *xmlstr, *str; + GtkTextIter iter; + int wrapped_lines; + int lines; + GdkRectangle oneline; + int height; + int pad_top, pad_inside, pad_bottom; + GtkTextIter start, end; + xmlnode *node; + + wrapped_lines = 1; + gtk_text_buffer_get_start_iter(buffer, &iter); + gtk_text_view_get_iter_location(GTK_TEXT_VIEW(console->entry), &iter, &oneline); + while (gtk_text_view_forward_display_line(GTK_TEXT_VIEW(console->entry), &iter)) + wrapped_lines++; + + lines = gtk_text_buffer_get_line_count(buffer); + + /* Show a maximum of 64 lines */ + lines = MIN(lines, 6); + wrapped_lines = MIN(wrapped_lines, 6); + + pad_top = gtk_text_view_get_pixels_above_lines(GTK_TEXT_VIEW(console->entry)); + pad_bottom = gtk_text_view_get_pixels_below_lines(GTK_TEXT_VIEW(console->entry)); + pad_inside = gtk_text_view_get_pixels_inside_wrap(GTK_TEXT_VIEW(console->entry)); + + height = (oneline.height + pad_top + pad_bottom) * lines; + height += (oneline.height + pad_inside) * (wrapped_lines - lines); + + gtk_widget_set_size_request(console->sw, -1, height+6); + + + + gtk_text_buffer_get_start_iter(buffer, &start); + gtk_text_buffer_get_end_iter(buffer, &end); + str = gtk_text_buffer_get_text(buffer, &start, &end, FALSE); + if (!str) + return; + xmlstr = g_strdup_printf("<xml>%s</xml>", str); + node = xmlnode_from_str(xmlstr, -1); + if (node) { + gtk_imhtml_clear_formatting(GTK_IMHTML(console->entry)); + } else { + gtk_imhtml_toggle_background(GTK_IMHTML(console->entry), "#ffcece"); + } + g_free(str); + g_free(xmlstr); + if (node) + xmlnode_free(node); +} + +static void iq_clicked_cb(GtkWidget *w, gpointer nul) +{ + GtkWidget *hbox, *to_entry, *label, *type_combo; + GtkSizeGroup *sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL); + GtkTextIter iter; + GtkTextBuffer *buffer; + const char *to; + int result; + char *stanza; + + GtkWidget *dialog = gtk_dialog_new_with_buttons("<iq/>", + GTK_WINDOW(console->window), + GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, + GTK_STOCK_CANCEL, + GTK_RESPONSE_REJECT, + GTK_STOCK_OK, + GTK_RESPONSE_ACCEPT, + NULL); + gtk_dialog_set_has_separator(GTK_DIALOG(dialog), FALSE); + gtk_dialog_set_default_response (GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT); + gtk_container_set_border_width(GTK_CONTAINER(dialog), 12); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("To:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + to_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (to_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), to_entry, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + label = gtk_label_new("Type:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + type_combo = gtk_combo_box_new_text(); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "get"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "set"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "result"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "error"); + gtk_combo_box_set_active(GTK_COMBO_BOX(type_combo), 0); + gtk_box_pack_start(GTK_BOX(hbox), type_combo, FALSE, FALSE, 0); + + gtk_widget_show_all(GTK_DIALOG(dialog)->vbox); + + result = gtk_dialog_run(GTK_DIALOG(dialog)); + if (result != GTK_RESPONSE_ACCEPT) { + gtk_widget_destroy(dialog); + return; + } + + to = gtk_entry_get_text(GTK_ENTRY(to_entry)); + + stanza = g_strdup_printf("<iq %s%s%s id='console%x' type='%s'></iq>", + to && *to ? "to='" : "", + to && *to ? to : "", + to && *to ? "'" : "", + g_random_int(), + gtk_combo_box_get_active_text(GTK_COMBO_BOX(type_combo))); + + buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(console->entry)); + gtk_text_buffer_set_text(buffer, stanza, -1); + gtk_text_buffer_get_iter_at_offset(buffer, &iter, strstr(stanza, "</iq>") - stanza); + gtk_text_buffer_place_cursor(buffer, &iter); + g_free(stanza); + + gtk_widget_destroy(dialog); + +} + +static void presence_clicked_cb(GtkWidget *w, gpointer nul) +{ + GtkWidget *hbox, + *to_entry, + *status_entry, + *priority_entry, + *label, + *show_combo, + *type_combo; + GtkSizeGroup *sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL); + GtkTextIter iter; + GtkTextBuffer *buffer; + const char *to, *type, *status, *show, *priority; + int result; + char *stanza; + + GtkWidget *dialog = gtk_dialog_new_with_buttons("<presence/>", + GTK_WINDOW(console->window), + GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, + GTK_STOCK_CANCEL, + GTK_RESPONSE_REJECT, + GTK_STOCK_OK, + GTK_RESPONSE_ACCEPT, + NULL); + gtk_dialog_set_has_separator(GTK_DIALOG(dialog), FALSE); + gtk_dialog_set_default_response (GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT); + gtk_container_set_border_width(GTK_CONTAINER(dialog), 12); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("To:"); + gtk_size_group_add_widget(sg, label); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + to_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (to_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), to_entry, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + label = gtk_label_new("Type:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + type_combo = gtk_combo_box_new_text(); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "default"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "unavailable"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "subscribe"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "unsubscribe"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "subscribed"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "unsubscribed"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "probe"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "error"); + gtk_combo_box_set_active(GTK_COMBO_BOX(type_combo), 0); + gtk_box_pack_start(GTK_BOX(hbox), type_combo, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + label = gtk_label_new("Show:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + show_combo = gtk_combo_box_new_text(); + gtk_combo_box_append_text(GTK_COMBO_BOX(show_combo), "default"); + gtk_combo_box_append_text(GTK_COMBO_BOX(show_combo), "away"); + gtk_combo_box_append_text(GTK_COMBO_BOX(show_combo), "dnd"); + gtk_combo_box_append_text(GTK_COMBO_BOX(show_combo), "xa"); + gtk_combo_box_append_text(GTK_COMBO_BOX(show_combo), "chat"); + + gtk_combo_box_set_active(GTK_COMBO_BOX(show_combo), 0); + gtk_box_pack_start(GTK_BOX(hbox), show_combo, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("Status:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + status_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (status_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), status_entry, FALSE, FALSE, 0); + + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("Priority:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + priority_entry = gtk_spin_button_new_with_range(-128, 127, 1); + gtk_spin_button_set_value(GTK_SPIN_BUTTON(priority_entry), 0); + gtk_box_pack_start(GTK_BOX(hbox), priority_entry, FALSE, FALSE, 0); + + + gtk_widget_show_all(GTK_DIALOG(dialog)->vbox); + + result = gtk_dialog_run(GTK_DIALOG(dialog)); + if (result != GTK_RESPONSE_ACCEPT) { + gtk_widget_destroy(dialog); + return; + } + + to = gtk_entry_get_text(GTK_ENTRY(to_entry)); + type = gtk_combo_box_get_active_text(GTK_COMBO_BOX(type_combo)); + if (!strcmp(type, "default")) + type = ""; + show = gtk_combo_box_get_active_text(GTK_COMBO_BOX(show_combo)); + if (!strcmp(show, "default")) + show = ""; + status = gtk_entry_get_text(GTK_ENTRY(status_entry)); + priority = gtk_entry_get_text(GTK_ENTRY(priority_entry)); + if (!strcmp(priority, "0")) + priority = ""; + + + + stanza = g_strdup_printf("<presence %s%s%s id='console%x' %s%s%s>" + "%s%s%s%s%s%s%s%s%s" + "</presence>", + *to ? "to='" : "", + *to ? to : "", + *to ? "'" : "", + g_random_int(), + + *type ? "type='" : "", + *type ? type : "", + *type ? "'" : "", + + *show ? "<show>" : "", + *show ? show : "", + *show ? "</show>" : "", + + *status ? "<status>" : "", + *status ? status : "", + *status ? "</status>" : "", + + *priority ? "<priority>" : "", + *priority ? priority : "", + *priority ? "</priority>" : ""); + + buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(console->entry)); + gtk_text_buffer_set_text(buffer, stanza, -1); + gtk_text_buffer_get_iter_at_offset(buffer, &iter, strstr(stanza, "</presence>") - stanza); + gtk_text_buffer_place_cursor(buffer, &iter); + g_free(stanza); + + gtk_widget_destroy(dialog); +} + +static void message_clicked_cb(GtkWidget *w, gpointer nul) +{ + GtkWidget *hbox, + *to_entry, + *body_entry, + *thread_entry, + *subject_entry, + *label, + *type_combo; + GtkSizeGroup *sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL); + GtkTextIter iter; + GtkTextBuffer *buffer; + const char *to, *body, *thread, *subject; + char *stanza; + int result; + + GtkWidget *dialog = gtk_dialog_new_with_buttons("<message/>", + GTK_WINDOW(console->window), + GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, + GTK_STOCK_CANCEL, + GTK_RESPONSE_REJECT, + GTK_STOCK_OK, + GTK_RESPONSE_ACCEPT, + NULL); + gtk_dialog_set_has_separator(GTK_DIALOG(dialog), FALSE); + gtk_dialog_set_default_response (GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT); + gtk_container_set_border_width(GTK_CONTAINER(dialog), 12); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("To:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + to_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (to_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), to_entry, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + label = gtk_label_new("Type:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + type_combo = gtk_combo_box_new_text(); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "chat"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "headline"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "groupchat"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "normal"); + gtk_combo_box_append_text(GTK_COMBO_BOX(type_combo), "error"); + gtk_combo_box_set_active(GTK_COMBO_BOX(type_combo), 0); + gtk_box_pack_start(GTK_BOX(hbox), type_combo, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("Body:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + body_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (body_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), body_entry, FALSE, FALSE, 0); + + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("Subject:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + subject_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (subject_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), subject_entry, FALSE, FALSE, 0); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), hbox, FALSE, FALSE, 0); + + label = gtk_label_new("Thread:"); + gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5); + gtk_size_group_add_widget(sg, label); + gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0); + + thread_entry = gtk_entry_new(); + gtk_entry_set_activates_default (GTK_ENTRY (thread_entry), TRUE); + gtk_box_pack_start(GTK_BOX(hbox), thread_entry, FALSE, FALSE, 0); + + gtk_widget_show_all(GTK_DIALOG(dialog)->vbox); + + result = gtk_dialog_run(GTK_DIALOG(dialog)); + if (result != GTK_RESPONSE_ACCEPT) { + gtk_widget_destroy(dialog); + return; + } + + to = gtk_entry_get_text(GTK_ENTRY(to_entry)); + body = gtk_entry_get_text(GTK_ENTRY(body_entry)); + thread = gtk_entry_get_text(GTK_ENTRY(thread_entry)); + subject = gtk_entry_get_text(GTK_ENTRY(subject_entry)); + + stanza = g_strdup_printf("<message %s%s%s id='console%x' type='%s'>" + "%s%s%s%s%s%s%s%s%s" + "</message>", + + *to ? "to='" : "", + *to ? to : "", + *to ? "'" : "", + g_random_int(), + gtk_combo_box_get_active_text(GTK_COMBO_BOX(type_combo)), + + *body ? "<body>" : "", + *body ? body : "", + *body ? "</body>" : "", + + *subject ? "<subject>" : "", + *subject ? subject : "", + *subject ? "</subject>" : "", + + *thread ? "<thread>" : "", + *thread ? thread : "", + *thread ? "</thread>" : ""); + + buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(console->entry)); + gtk_text_buffer_set_text(buffer, stanza, -1); + gtk_text_buffer_get_iter_at_offset(buffer, &iter, strstr(stanza, "</message>") - stanza); + gtk_text_buffer_place_cursor(buffer, &iter); + g_free(stanza); + + gtk_widget_destroy(dialog); +} + +static void +signed_on_cb(GaimConnection *gc) +{ + if (!console) + return; + + gtk_combo_box_append_text(GTK_COMBO_BOX(console->dropdown), gaim_account_get_username(gc->account)); + console->accounts = g_list_append(console->accounts, gc); + console->count++; + + if (console->count > 1) + gtk_widget_show_all(console->hbox); +} + +static void +signed_off_cb(GaimConnection *gc) +{ + int i = 0; + GList *l; + + if (!console) + return; + + l = console->accounts; + while (l) { + GaimConnection *g = l->data; + if (gc == g) + break; + i++; + l = l->next; + } + + if (l == NULL) + return; + + gtk_combo_box_remove_text(GTK_COMBO_BOX(console->dropdown), i); + console->accounts = g_list_remove(console->accounts, gc); + printf("%s\n", gaim_account_get_username(gc->account)); + console->count--; + + if (gc == console->gc) { + console->gc = NULL; + gtk_imhtml_append_text(GTK_IMHTML(console->imhtml), + _("<font color='#777777'>Logged out.</font>"), 0); + } +} + +static gboolean +plugin_load(GaimPlugin *plugin) +{ + GaimPlugin *jabber; + + jabber = gaim_find_prpl("prpl-jabber"); + if (!jabber) + return FALSE; + + xmpp_console_handle = plugin; + gaim_signal_connect(jabber, "jabber-receiving-xmlnode", xmpp_console_handle, + GAIM_CALLBACK(xmlnode_received_cb), NULL); + gaim_signal_connect(jabber, "jabber-sending-text", xmpp_console_handle, + GAIM_CALLBACK(xmlnode_sent_cb), NULL); + gaim_signal_connect(gaim_connections_get_handle(), "signed-on", + plugin, GAIM_CALLBACK(signed_on_cb), NULL); + gaim_signal_connect(gaim_connections_get_handle(), "signed-off", + plugin, GAIM_CALLBACK(signed_off_cb), NULL); + + return TRUE; +} + +static gboolean +plugin_unload(GaimPlugin *plugin) +{ + gtk_widget_destroy(console->window); + console->window = NULL; + return TRUE; +} + +static void +console_destroy(GtkObject *window, gpointer nul) +{ + g_list_free(console->accounts); + g_free(console); + console = NULL; +} + +static void +dropdown_changed_cb(GtkComboBox *widget, gpointer nul) +{ + GaimAccount *account; + + if (!console) + return; + + account = gaim_accounts_find(gtk_combo_box_get_active_text(GTK_COMBO_BOX(console->dropdown)), + "prpl-jabber"); + if (!account || !account->gc) + return; + + console->gc = account->gc; + gtk_imhtml_clear(GTK_IMHTML(console->imhtml)); +} + +static void +create_console() +{ + GtkWidget *vbox = gtk_vbox_new(FALSE, 6); + GtkWidget *sw = gtk_scrolled_window_new(NULL, NULL); + GtkWidget *label; + GtkTextBuffer *buffer; + GtkWidget *toolbar; + GtkToolItem *button; + GList *connections; + + if (console) { + gtk_window_present(GTK_WINDOW(console->window)); + return; + } + + console = g_new0(XmppConsole, 1); + + console->window = gtk_window_new(GTK_WINDOW_TOPLEVEL); + gtk_window_set_title(GTK_WINDOW(console->window), _("XMPP Console")); + g_signal_connect(G_OBJECT(console->window), "destroy", G_CALLBACK(console_destroy), NULL); + gtk_container_set_border_width(GTK_CONTAINER(console->window), 12); + gtk_window_set_default_size(GTK_WINDOW(console->window), 580, 400); + gtk_container_add(GTK_CONTAINER(console->window), vbox); + + console->hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(vbox), console->hbox, FALSE, FALSE, 0); + label = gtk_label_new(_("Account: ")); + gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5); + gtk_box_pack_start(GTK_BOX(console->hbox), label, FALSE, FALSE, 0); + console->dropdown = gtk_combo_box_new_text(); + for (connections = gaim_connections_get_all(); connections; connections = connections->next) { + GaimConnection *gc = connections->data; + if (!strcmp(gaim_account_get_protocol_id(gaim_connection_get_account(gc)), "prpl-jabber")) { + console->count++; + console->accounts = g_list_append(console->accounts, gc); + gtk_combo_box_append_text(GTK_COMBO_BOX(console->dropdown), + gaim_account_get_username(gaim_connection_get_account(gc))); + if (!console->gc) + console->gc = gc; + } + } + gtk_combo_box_set_active(GTK_COMBO_BOX(console->dropdown),0); + gtk_box_pack_start(GTK_BOX(console->hbox), console->dropdown, TRUE, TRUE, 0); + g_signal_connect(G_OBJECT(console->dropdown), "changed", G_CALLBACK(dropdown_changed_cb), NULL); + + gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_ETCHED_IN); + gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), + GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + + console->imhtml = gtk_imhtml_new(NULL, NULL); + gtk_box_pack_start(GTK_BOX(vbox), sw, TRUE, TRUE, 0); + if (console->count == 0) + gtk_imhtml_append_text(GTK_IMHTML(console->imhtml), + _("<font color='#777777'>Not connected to XMPP</font>"), 0); + gtk_container_add(GTK_CONTAINER(sw), console->imhtml); + + toolbar = gtk_toolbar_new(); + button = gtk_tool_button_new(NULL, "<iq/>"); + g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(iq_clicked_cb), NULL); + gtk_container_add(GTK_CONTAINER(toolbar), GTK_WIDGET(button)); + + button = gtk_tool_button_new(NULL, "<presence/>"); + g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(presence_clicked_cb), NULL); + gtk_container_add(GTK_CONTAINER(toolbar), GTK_WIDGET(button)); + + button = gtk_tool_button_new(NULL, "<message/>"); + g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(message_clicked_cb), NULL); + gtk_container_add(GTK_CONTAINER(toolbar), GTK_WIDGET(button)); + + gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 0); + + sw = gtk_scrolled_window_new(NULL, NULL); + gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_ETCHED_IN); + gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), + GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); + + console->entry = gtk_imhtml_new(NULL, NULL); + gtk_imhtml_set_whole_buffer_formatting_only(GTK_IMHTML(console->entry), TRUE); + g_signal_connect(G_OBJECT(console->entry),"message_send", G_CALLBACK(message_send_cb), console); + + gtk_box_pack_start(GTK_BOX(vbox), sw, FALSE, FALSE, 0); + gtk_container_add(GTK_CONTAINER(sw), console->entry); + gtk_imhtml_set_editable(GTK_IMHTML(console->entry), TRUE); + buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(console->entry)); + g_signal_connect(G_OBJECT(buffer), "changed", G_CALLBACK(entry_changed_cb), NULL); + console->sw = sw; + entry_changed_cb(buffer, NULL); + + gtk_widget_show_all(console->window); + if (console->count < 2) + gtk_widget_hide(console->hbox); +} + +static GList * +actions(GaimPlugin *plugin, gpointer context) +{ + GList *l = NULL; + GaimPluginAction *act = NULL; + + act = gaim_plugin_action_new(_("XMPP Console"), create_console); + l = g_list_append(l, act); + + return l; +} + + +static GaimPluginInfo info = +{ + GAIM_PLUGIN_MAGIC, + GAIM_MAJOR_VERSION, + GAIM_MINOR_VERSION, + GAIM_PLUGIN_STANDARD, /**< type */ + GAIM_GTK_PLUGIN_TYPE, /**< ui_requirement */ + 0, /**< flags */ + NULL, /**< dependencies */ + GAIM_PRIORITY_DEFAULT, /**< priority */ + + "gtk-xmpp", /**< id */ + N_("XMPP Console"), /**< name */ + VERSION, /**< version */ + /** summary */ + N_("Send and receive raw XMPP stanzas."), + /** description */ + N_("This plugin is useful for debbuging XMPP servers or clients."), + "Sean Egan <seanegan@gmail.com>", /**< author */ + GAIM_WEBSITE, /**< homepage */ + + plugin_load, /**< load */ + plugin_unload, /**< unload */ + NULL, /**< destroy */ + + NULL, /**< ui_info */ + NULL, /**< extra_info */ + NULL, + actions +}; + +static void +init_plugin(GaimPlugin *plugin) +{ +} + +GAIM_INIT_PLUGIN(interval, init_plugin, info)
--- a/libgaim/protocols/jabber/jabber.c Wed Jan 17 06:29:23 2007 +0000 +++ b/libgaim/protocols/jabber/jabber.c Wed Jan 17 08:58:26 2007 +0000 @@ -257,7 +257,9 @@ /* If we've got a security layer, we need to encode the data, * splitting it on the maximum buffer length negotiated */ - gaim_signal_emit(my_protocol, "jabber-sending-text", js->gc, data); + gaim_signal_emit(my_protocol, "jabber-sending-text", js->gc, &data); + if (data == NULL) + return; #ifdef HAVE_CYRUS_SASL if (js->sasl_maxbuf>0) { @@ -1925,7 +1927,7 @@ gaim_signal_register(plugin, "jabber-sending-text", gaim_marshal_VOID__POINTER_POINTER, NULL, 2, gaim_value_new(GAIM_TYPE_SUBTYPE, GAIM_SUBTYPE_CONNECTION), - gaim_value_new(GAIM_TYPE_STRING)); + gaim_value_new_outgoing(GAIM_TYPE_STRING)); return TRUE;
--- a/libgaim/xmlnode.c Wed Jan 17 06:29:23 2007 +0000 +++ b/libgaim/xmlnode.c Wed Jan 17 08:58:26 2007 +0000 @@ -441,6 +441,7 @@ struct _xmlnode_parser_data { xmlnode *current; + gboolean error; }; static void @@ -453,7 +454,7 @@ xmlnode *node; int i; - if(!element_name) { + if(!element_name || xpd->error) { return; } else { if(xpd->current) @@ -486,7 +487,7 @@ { struct _xmlnode_parser_data *xpd = user_data; - if(!element_name || !xpd->current) + if(!element_name || !xpd->current || xpd->error) return; if(xpd->current->parent) { @@ -500,15 +501,22 @@ { struct _xmlnode_parser_data *xpd = user_data; - if(!xpd->current) + if(!xpd->current || xpd->error) return; - + if(!text || !text_len) return; xmlnode_insert_data(xpd->current, (const char*) text, text_len); } +static void +xmlnode_parser_error_libxml(void *user_data, const char *msg, ...) +{ + struct _xmlnode_parser_data *xpd = user_data; + xpd->error = TRUE; +} + static xmlSAXHandler xmlnode_parser_libxml = { .internalSubset = NULL, .isStandalone = NULL, @@ -532,7 +540,7 @@ .processingInstruction = NULL, .comment = NULL, .warning = NULL, - .error = NULL, + .error = xmlnode_parser_error_libxml, .fatalError = NULL, .getParameterEntity = NULL, .cdataBlock = NULL, @@ -564,6 +572,12 @@ xpd->current = NULL; } ret = xpd->current; + if (xpd->error) { + ret = NULL; + if (xpd->current) + xmlnode_free(xpd->current); + } + g_free(xpd); return ret; }