diff pidgin/gtkplugin.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 d75099d2567e
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/pidgin/gtkplugin.c	Sat Jan 20 02:32:10 2007 +0000
@@ -0,0 +1,608 @@
+/**
+ * @file gtkplugin.c GTK+ Plugins support
+ * @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 "gtkplugin.h"
+#include "gtkpluginpref.h"
+#include "gtkutils.h"
+#include "debug.h"
+#include "prefs.h"
+#include "request.h"
+
+#include <string.h>
+
+#define GAIM_RESPONSE_CONFIGURE 98121
+
+static void plugin_toggled_stage_two(GaimPlugin *plug, GtkTreeModel *model,
+                                  GtkTreeIter *iter, gboolean unload);
+
+static GtkWidget *expander = NULL;
+static GtkWidget *plugin_dialog = NULL;
+static GtkWidget *plugin_details = NULL;
+static GtkWidget *pref_button = NULL;
+static GHashTable *plugin_pref_dialogs = NULL;
+
+GtkWidget *
+gaim_gtk_plugin_get_config_frame(GaimPlugin *plugin)
+{
+	GtkWidget *config = NULL;
+
+	g_return_val_if_fail(plugin != NULL, NULL);
+
+	if (GAIM_IS_GTK_PLUGIN(plugin) && plugin->info->ui_info
+		&& GAIM_GTK_PLUGIN_UI_INFO(plugin)->get_config_frame)
+	{
+		GaimGtkPluginUiInfo *ui_info;
+
+		ui_info = GAIM_GTK_PLUGIN_UI_INFO(plugin);
+
+		config = ui_info->get_config_frame(plugin);
+
+		if (plugin->info->prefs_info
+			&& plugin->info->prefs_info->get_plugin_pref_frame)
+		{
+			gaim_debug_warning("gtkplugin",
+				"Plugin %s contains both, ui_info and "
+				"prefs_info preferences; prefs_info will be "
+				"ignored.", plugin->info->name);
+		}
+	}
+
+	if (config == NULL && plugin->info->prefs_info
+		&& plugin->info->prefs_info->get_plugin_pref_frame)
+	{
+		GaimPluginPrefFrame *frame;
+
+		frame = plugin->info->prefs_info->get_plugin_pref_frame(plugin);
+
+		config = gaim_gtk_plugin_pref_create_frame(frame);
+
+		/* XXX According to bug #1407047 this broke saving pluging preferences, I'll look at fixing it correctly later.
+		gaim_plugin_pref_frame_destroy(frame);
+		*/
+	}
+
+	return config;
+}
+
+void
+gaim_gtk_plugins_save(void)
+{
+	gaim_plugins_save_loaded("/gaim/gtk/plugins/loaded");
+}
+
+static void
+update_plugin_list(void *data)
+{
+	GtkListStore *ls = GTK_LIST_STORE(data);
+	GtkTreeIter iter;
+	GList *probes;
+	GaimPlugin *plug;
+
+	gtk_list_store_clear(ls);
+	gaim_plugins_probe(G_MODULE_SUFFIX);
+
+	for (probes = gaim_plugins_get_all();
+		 probes != NULL;
+		 probes = probes->next)
+	{
+		char *name;
+		char *version;
+		char *summary;
+		char *desc;
+		plug = probes->data;
+
+		if (plug->info->type == GAIM_PLUGIN_LOADER) {
+			GList *cur;
+			for (cur = GAIM_PLUGIN_LOADER_INFO(plug)->exts; cur != NULL;
+					 cur = cur->next)
+				gaim_plugins_probe(cur->data);
+			continue;
+		} else if (plug->info->type != GAIM_PLUGIN_STANDARD ||
+			(plug->info->flags & GAIM_PLUGIN_FLAG_INVISIBLE)) {
+			continue;
+		}
+
+		gtk_list_store_append (ls, &iter);
+
+		name = g_markup_escape_text(plug->info->name ? _(plug->info->name) : g_basename(plug->path), -1);
+		version = g_markup_escape_text(plug->info->version, -1);
+		summary = g_markup_escape_text(_(plug->info->summary), -1);
+
+		desc = g_strdup_printf("<b>%s</b> %s\n%s", name,
+				       version,
+				       summary);
+		g_free(name);
+		g_free(version);
+		g_free(summary);
+
+		gtk_list_store_set(ls, &iter,
+				   0, gaim_plugin_is_loaded(plug),
+				   1, desc,
+				   2, plug,
+				   3, gaim_plugin_is_unloadable(plug),
+				   -1);
+		g_free(desc);
+	}
+}
+
+static void plugin_loading_common(GaimPlugin *plugin, GtkTreeView *view, gboolean loaded)
+{
+	GtkTreeIter iter;
+	GtkTreeModel *model = gtk_tree_view_get_model(view);
+
+	if (gtk_tree_model_get_iter_first(model, &iter)) {
+		do {
+			GaimPlugin *plug;
+			GtkTreeSelection *sel;
+
+			gtk_tree_model_get(model, &iter, 2, &plug, -1);
+
+			if (plug != plugin)
+				continue;
+
+			gtk_list_store_set(GTK_LIST_STORE(model), &iter, 0, loaded, -1);
+
+			/* If the loaded/unloaded plugin is the selected row,
+			 * update the pref_button. */
+			sel = gtk_tree_view_get_selection(view);
+			if (gtk_tree_selection_get_selected(sel, &model, &iter))
+			{
+				gtk_tree_model_get(model, &iter, 2, &plug, -1);
+				if (plug == plugin)
+				{
+					gtk_widget_set_sensitive(pref_button,
+						loaded
+						&& ((GAIM_IS_GTK_PLUGIN(plug) && plug->info->ui_info
+							&& GAIM_GTK_PLUGIN_UI_INFO(plug)->get_config_frame)
+						 || (plug->info->prefs_info
+							&& plug->info->prefs_info->get_plugin_pref_frame)));
+				}
+			}
+
+			break;
+		} while (gtk_tree_model_iter_next(model, &iter));
+	}
+}
+
+static void plugin_load_cb(GaimPlugin *plugin, gpointer data)
+{
+	GtkTreeView *view = (GtkTreeView *)data;
+	plugin_loading_common(plugin, view, TRUE);
+}
+
+static void plugin_unload_cb(GaimPlugin *plugin, gpointer data)
+{
+	GtkTreeView *view = (GtkTreeView *)data;
+	plugin_loading_common(plugin, view, FALSE);
+}
+
+static void pref_dialog_response_cb(GtkWidget *d, int response, GaimPlugin *plug)
+{
+	switch (response) {
+	case GTK_RESPONSE_CLOSE:
+	case GTK_RESPONSE_DELETE_EVENT:
+		g_hash_table_remove(plugin_pref_dialogs, plug);
+		if (g_hash_table_size(plugin_pref_dialogs) == 0) {
+			g_hash_table_destroy(plugin_pref_dialogs);
+			plugin_pref_dialogs = NULL;
+		}
+		gtk_widget_destroy(d);
+		break;
+	}
+}
+
+static void plugin_unload_confirm_cb(gpointer *data)
+{
+	GaimPlugin *plugin = (GaimPlugin *)data[0];
+	GtkTreeModel *model = (GtkTreeModel *)data[1];
+	GtkTreeIter *iter = (GtkTreeIter *)data[2];
+
+	plugin_toggled_stage_two(plugin, model, iter, TRUE);
+
+	g_free(data);
+}
+
+static void plugin_toggled(GtkCellRendererToggle *cell, gchar *pth, gpointer data)
+{
+	GtkTreeModel *model = (GtkTreeModel *)data;
+	GtkTreeIter *iter = g_new(GtkTreeIter, 1);
+	GtkTreePath *path = gtk_tree_path_new_from_string(pth);
+	GaimPlugin *plug;
+	GtkWidget *dialog = NULL;
+
+	gtk_tree_model_get_iter(model, iter, path);
+	gtk_tree_path_free(path);
+	gtk_tree_model_get(model, iter, 2, &plug, -1);
+
+	/* Apparently, GTK+ won't honor the sensitive flag on cell renderers for booleans. */
+	if (gaim_plugin_is_unloadable(plug))
+	{
+		g_free(iter);
+		return;
+	}
+
+	if (!gaim_plugin_is_loaded(plug))
+	{
+		gaim_gtk_set_cursor(plugin_dialog, GDK_WATCH);
+
+		gaim_plugin_load(plug);
+		plugin_toggled_stage_two(plug, model, iter, FALSE);
+
+		gaim_gtk_clear_cursor(plugin_dialog);
+	}
+	else
+	{
+		if (plugin_pref_dialogs != NULL &&
+			(dialog = g_hash_table_lookup(plugin_pref_dialogs, plug)))
+			pref_dialog_response_cb(dialog, GTK_RESPONSE_DELETE_EVENT, plug);
+
+		if (plug->dependent_plugins != NULL)
+		{
+			GString *tmp = g_string_new(_("The following plugins will be unloaded."));
+			GList *l;
+			gpointer *cb_data;
+
+			for (l = plug->dependent_plugins ; l != NULL ; l = l->next)
+			{
+				const char *dep_name = (const char *)l->data;
+				GaimPlugin *dep_plugin = gaim_plugins_find_with_id(dep_name);
+				g_return_if_fail(dep_plugin != NULL);
+
+				g_string_append_printf(tmp, "\n\t%s\n", _(dep_plugin->info->name));
+			}
+
+			cb_data = g_new(gpointer, 3);
+			cb_data[0] = plug;
+			cb_data[1] = model;
+			cb_data[2] = iter;
+
+			gaim_request_action(plugin_dialog, NULL,
+			                    _("Multiple plugins will be unloaded."),
+			                    tmp->str, 0, cb_data, 2,
+			                    _("Unload Plugins"), G_CALLBACK(plugin_unload_confirm_cb),
+			                    _("Cancel"), g_free);
+			g_string_free(tmp, TRUE);
+		}
+		else
+			plugin_toggled_stage_two(plug, model, iter, TRUE);
+	}
+}
+
+static void plugin_toggled_stage_two(GaimPlugin *plug, GtkTreeModel *model, GtkTreeIter *iter, gboolean unload)
+{
+	gchar *name = NULL;
+	gchar *description = NULL;
+
+	if (unload)
+	{
+		gaim_gtk_set_cursor(plugin_dialog, GDK_WATCH);
+
+		gaim_plugin_unload(plug);
+
+		gaim_gtk_clear_cursor(plugin_dialog);
+	}
+
+	gtk_widget_set_sensitive(pref_button,
+		gaim_plugin_is_loaded(plug)
+		&& ((GAIM_IS_GTK_PLUGIN(plug) && plug->info->ui_info
+			&& GAIM_GTK_PLUGIN_UI_INFO(plug)->get_config_frame)
+		 || (plug->info->prefs_info
+			&& plug->info->prefs_info->get_plugin_pref_frame)));
+
+	name = g_markup_escape_text(_(plug->info->name), -1);
+	description = g_markup_escape_text(_(plug->info->description), -1);
+
+	if (plug->error != NULL) {
+		gchar *error = g_markup_escape_text(plug->error, -1);
+		gchar *desc;
+		gchar *text = g_strdup_printf(
+				   "<span size=\"larger\">%s %s</span>\n\n"
+				   "<span weight=\"bold\" color=\"red\">%s</span>\n\n"
+				   "%s",
+				   name, plug->info->version, error, description);
+		desc = g_strdup_printf("<b>%s</b> %s\n<span weight=\"bold\" color=\"red\"%s</span>",
+			       plug->info->name, plug->info->version, error);
+		gtk_list_store_set(GTK_LIST_STORE (model), iter,
+				   1, desc,
+				   -1);
+		g_free(desc);
+		g_free(error);
+		gtk_label_set_markup(GTK_LABEL(plugin_details), text);
+		g_free(text);
+	}
+	g_free(name);
+	g_free(description);
+
+
+	gtk_list_store_set(GTK_LIST_STORE (model), iter,
+	                   0, gaim_plugin_is_loaded(plug),
+	                   -1);
+	g_free(iter);
+
+	gaim_gtk_plugins_save();
+}
+
+static gboolean ensure_plugin_visible(void *data)
+{
+	GtkTreeSelection *sel = GTK_TREE_SELECTION(data);
+	GtkTreeView *tv = gtk_tree_selection_get_tree_view(sel);
+	GtkTreeModel *model = gtk_tree_view_get_model(tv);
+	GtkTreePath *path;
+	GtkTreeIter iter;
+	if (!gtk_tree_selection_get_selected (sel, &model, &iter))
+		return FALSE;
+	path = gtk_tree_model_get_path(model, &iter);
+	gtk_tree_view_scroll_to_cell(gtk_tree_selection_get_tree_view(sel), path, NULL, FALSE, 0, 0);
+	gtk_tree_path_free(path);
+	return FALSE;
+}
+
+static void prefs_plugin_sel (GtkTreeSelection *sel, GtkTreeModel *model)
+{
+	gchar *buf, *pname, *pdesc, *pauth, *pweb;
+	GtkTreeIter  iter;
+	GValue val;
+	GaimPlugin *plug;
+
+	if (!gtk_tree_selection_get_selected (sel, &model, &iter))
+	{
+		/* Clear the old plugin details */
+		gtk_label_set_markup(GTK_LABEL(plugin_details), "");
+		gtk_widget_set_sensitive(pref_button, FALSE);
+
+		/* Collapse and disable the expander widget */
+		gtk_expander_set_expanded(GTK_EXPANDER(expander), FALSE);
+		gtk_widget_set_sensitive(expander, FALSE);
+
+		return;
+	}
+
+	gtk_widget_set_sensitive(expander, TRUE);
+
+	val.g_type = 0;
+	gtk_tree_model_get_value (model, &iter, 2, &val);
+	plug = g_value_get_pointer(&val);
+
+	pname = g_markup_escape_text(_(plug->info->name), -1);
+	pdesc = (plug->info->description) ?
+			g_markup_escape_text(_(plug->info->description), -1) : NULL;
+	pauth = (plug->info->author) ?
+			g_markup_escape_text(_(plug->info->author), -1) : NULL;
+	pweb = (plug->info->homepage) ?
+		   g_markup_escape_text(_(plug->info->homepage), -1) : NULL;
+	buf = g_strdup_printf(
+		   _("%s%s"
+		     "<span weight=\"bold\">Written by:</span>\t%s\n"
+		     "<span weight=\"bold\">Website:</span>\t\t%s\n"
+		     "<span weight=\"bold\">Filename:</span>\t\t%s"),
+		   pdesc ? pdesc : "", pdesc ? "\n\n" : "",
+		   pauth ? pauth : "", pweb ? pweb : "", plug->path);
+
+	if (plug->error != NULL)
+	{
+		char *tmp = g_strdup_printf(
+			_("%s\n"
+			  "<span foreground=\"#ff0000\" weight=\"bold\">"
+			  "Error: %s\n"
+			  "Check the plugin website for an update."
+			  "</span>"),
+			buf, plug->error);
+		g_free(buf);
+		buf = tmp;
+	}
+
+	gtk_widget_set_sensitive(pref_button,
+		gaim_plugin_is_loaded(plug)
+		&& ((GAIM_IS_GTK_PLUGIN(plug) && plug->info->ui_info
+			&& GAIM_GTK_PLUGIN_UI_INFO(plug)->get_config_frame)
+		 || (plug->info->prefs_info
+			&& plug->info->prefs_info->get_plugin_pref_frame)));
+
+	gtk_label_set_markup(GTK_LABEL(plugin_details), buf);
+
+	/* Make sure the selected plugin is still visible */
+	g_idle_add(ensure_plugin_visible, sel);
+
+
+	g_value_unset(&val);
+	g_free(buf);
+	g_free(pname);
+	g_free(pdesc);
+	g_free(pauth);
+	g_free(pweb);
+}
+
+static void plugin_dialog_response_cb(GtkWidget *d, int response, GtkTreeSelection *sel)
+{
+	GaimPlugin *plug;
+	GtkWidget *dialog, *box;
+	GtkTreeModel *model;
+	GValue val;
+	GtkTreeIter iter;
+
+	switch (response) {
+	case GTK_RESPONSE_CLOSE:
+	case GTK_RESPONSE_DELETE_EVENT:
+		gaim_request_close_with_handle(plugin_dialog);
+		gaim_signals_disconnect_by_handle(plugin_dialog);
+		gtk_widget_destroy(d);
+		if (plugin_pref_dialogs != NULL) {
+			g_hash_table_destroy(plugin_pref_dialogs);
+			plugin_pref_dialogs = NULL;
+		}
+		plugin_dialog = NULL;
+		break;
+	case GAIM_RESPONSE_CONFIGURE:
+		if (! gtk_tree_selection_get_selected (sel, &model, &iter))
+			return;
+		val.g_type = 0;
+		gtk_tree_model_get_value(model, &iter, 2, &val);
+		plug = g_value_get_pointer(&val);
+		if (plug == NULL)
+			break;
+		if (plugin_pref_dialogs != NULL &&
+			g_hash_table_lookup(plugin_pref_dialogs, plug))
+			break;
+		box = gaim_gtk_plugin_get_config_frame(plug);
+		if (box == NULL)
+			break;
+
+		dialog = gtk_dialog_new_with_buttons(GAIM_ALERT_TITLE, GTK_WINDOW(d),
+						     GTK_DIALOG_NO_SEPARATOR | GTK_DIALOG_DESTROY_WITH_PARENT,
+						     GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
+						     NULL);
+		if (plugin_pref_dialogs == NULL)
+			plugin_pref_dialogs = g_hash_table_new(NULL, NULL);
+
+		g_hash_table_insert(plugin_pref_dialogs, plug, dialog);
+
+		g_signal_connect(G_OBJECT(dialog), "response", G_CALLBACK(pref_dialog_response_cb), plug);
+		gtk_container_add(GTK_CONTAINER(GTK_DIALOG(dialog)->vbox), box);
+		gtk_window_set_role(GTK_WINDOW(dialog), "plugin_config");
+		gtk_window_set_title(GTK_WINDOW(dialog), _(gaim_plugin_get_name(plug)));
+		gtk_widget_show_all(dialog);
+		g_value_unset(&val);
+		break;
+	}
+}
+
+static void
+show_plugin_prefs_cb(GtkTreeView *view, GtkTreePath *path, GtkTreeViewColumn *column, GtkWidget *dialog)
+{
+	GtkTreeSelection *sel;
+	GtkTreeIter iter;
+	GaimPlugin *plugin;
+	GtkTreeModel *model;
+
+	sel = gtk_tree_view_get_selection(view);
+
+	if (!gtk_tree_selection_get_selected(sel, &model, &iter))
+		return;
+
+	gtk_tree_model_get(model, &iter, 2, &plugin, -1);
+
+	if (!gaim_plugin_is_loaded(plugin))
+		return;
+
+	/* Now show the pref-dialog for the plugin */
+	plugin_dialog_response_cb(dialog, GAIM_RESPONSE_CONFIGURE, sel);
+}
+
+void gaim_gtk_plugin_dialog_show()
+{
+	GtkWidget *sw;
+	GtkWidget *event_view;
+	GtkListStore *ls;
+	GtkCellRenderer *rend, *rendt;
+	GtkTreeViewColumn *col;
+	GtkTreeSelection *sel;
+
+	if (plugin_dialog != NULL) {
+		gtk_window_present(GTK_WINDOW(plugin_dialog));
+		return;
+	}
+
+	plugin_dialog = gtk_dialog_new_with_buttons(_("Plugins"),
+						    NULL,
+						    GTK_DIALOG_NO_SEPARATOR,
+						    NULL);
+	pref_button = gtk_dialog_add_button(GTK_DIALOG(plugin_dialog),
+						_("Configure Pl_ugin"), GAIM_RESPONSE_CONFIGURE);
+	gtk_dialog_add_button(GTK_DIALOG(plugin_dialog),
+						GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE);
+	gtk_widget_set_sensitive(pref_button, FALSE);
+	gtk_window_set_role(GTK_WINDOW(plugin_dialog), "plugins");
+
+	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(GTK_DIALOG(plugin_dialog)->vbox), sw, TRUE, TRUE, 0);
+
+	ls = gtk_list_store_new(4, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_POINTER, G_TYPE_BOOLEAN);
+	gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(ls),
+					     1, GTK_SORT_ASCENDING);
+
+	update_plugin_list(ls);
+
+	event_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(ls));
+
+	gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(event_view), TRUE);
+
+	g_signal_connect(G_OBJECT(event_view), "row-activated",
+				G_CALLBACK(show_plugin_prefs_cb), plugin_dialog);
+
+	gaim_signal_connect(gaim_plugins_get_handle(), "plugin-load", plugin_dialog,
+	                    GAIM_CALLBACK(plugin_load_cb), event_view);
+	gaim_signal_connect(gaim_plugins_get_handle(), "plugin-unload", plugin_dialog,
+	                    GAIM_CALLBACK(plugin_unload_cb), event_view);
+
+	rend = gtk_cell_renderer_toggle_new();
+	sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (event_view));
+
+	col = gtk_tree_view_column_new_with_attributes (_("Enabled"),
+							rend,
+							"active", 0,
+							NULL);
+	gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
+	gtk_tree_view_column_set_sort_column_id(col, 0);
+	g_signal_connect(G_OBJECT(rend), "toggled",
+			 G_CALLBACK(plugin_toggled), ls);
+
+	rendt = gtk_cell_renderer_text_new();
+	g_object_set(rendt,
+		     "foreground", "#c0c0c0",
+		     NULL);
+	col = gtk_tree_view_column_new_with_attributes (_("Name"),
+							rendt,
+							"markup", 1,
+							"foreground-set", 3,
+							NULL);
+#if GTK_CHECK_VERSION(2,6,0)
+	gtk_tree_view_column_set_expand (col, TRUE);
+	g_object_set(rendt, "ellipsize", PANGO_ELLIPSIZE_END, NULL);
+#endif
+	gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
+	gtk_tree_view_column_set_sort_column_id(col, 1);
+	g_object_unref(G_OBJECT(ls));
+	gtk_container_add(GTK_CONTAINER(sw), event_view);
+	gtk_tree_view_set_search_column(GTK_TREE_VIEW(event_view), 1);
+	gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(event_view),
+				gaim_gtk_tree_view_search_equal_func, NULL, NULL);
+
+	expander = gtk_expander_new(_("<b>Plugin Details</b>"));
+	gtk_expander_set_use_markup(GTK_EXPANDER(expander), TRUE);
+	plugin_details = gtk_label_new(NULL);
+	gtk_label_set_line_wrap(GTK_LABEL(plugin_details), TRUE);
+	gtk_container_add(GTK_CONTAINER(expander), plugin_details);
+	gtk_widget_set_sensitive(expander, FALSE);
+	gtk_box_pack_start(GTK_BOX(GTK_DIALOG(plugin_dialog)->vbox), expander, FALSE, FALSE, 0);
+
+	g_signal_connect (G_OBJECT (sel), "changed", G_CALLBACK (prefs_plugin_sel), NULL);
+	g_signal_connect(G_OBJECT(plugin_dialog), "response", G_CALLBACK(plugin_dialog_response_cb), sel);
+	gtk_window_set_default_size(GTK_WINDOW(plugin_dialog), 430, 430);
+	gtk_widget_show_all(plugin_dialog);
+}