2006-02-21 [wwp] 2.0.0cvs69
[claws.git] / src / gtk / pluginwindow.c
index 6917022db710a5dd2d8f8b42ee53074db0606a4e..86b60d5c3133955b1d1f5111ed75836bc506086e 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
- * Copyright (C) 1999-2002 Hiroyuki Yamamoto and the Sylpheed-Claws Team
+ * Copyright (C) 1999-2006 Hiroyuki Yamamoto and the Sylpheed-Claws Team
  *
  * 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
  *
  * 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.
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  */
 
 #ifdef HAVE_CONFIG_H
 #  include "config.h"
 #endif
 
+#include <glib.h>
+#include <glib/gi18n.h>
 #include <string.h>
 
 #include <gtk/gtk.h>
 #include <gdk/gdkkeysyms.h>
 
-#include "intl.h"
 #include "plugin.h"
 
 #include "filesel.h"
 #include "alertpanel.h"
+#include "prefs_common.h"
 #include "../inc.h"
 
+enum {
+       PLUGINWINDOW_NAME,              /*<! plugin name */
+       PLUGINWINDOW_DATA,              /*<! Plugin pointer */
+       PLUGINWINDOW_STYLE,             /*<! italic if error */
+       N_PLUGINWINDOW_COLUMNS
+};
+
 typedef struct _PluginWindow
 {
        GtkWidget *window;
-       GtkWidget *plugin_list;
+       GtkWidget *plugin_list_view;
        GtkWidget *plugin_desc;
        GtkWidget *unload_btn;
 
        Plugin *selected_plugin;
 } PluginWindow;
 
+static GtkListStore* pluginwindow_create_data_store    (void);
+static GtkWidget *pluginwindow_list_view_create                (PluginWindow *pluginwindow);
+static void pluginwindow_create_list_view_columns      (GtkWidget *list_view);
+static gboolean pluginwindow_selected                  (GtkTreeSelection *selector,
+                                                        GtkTreeModel *model, 
+                                                        GtkTreePath *path,
+                                                        gboolean currently_selected,
+                                                        gpointer data);
+
 static void close_cb(GtkButton *button, PluginWindow *pluginwindow)
 {
        gtk_widget_destroy(pluginwindow->window);
@@ -53,17 +71,26 @@ static void close_cb(GtkButton *button, PluginWindow *pluginwindow)
 
 static void set_plugin_list(PluginWindow *pluginwindow)
 {
-       GSList *plugins, *cur;
-       gchar *text[1];
-       gint row;
-       GtkCList *clist = GTK_CLIST(pluginwindow->plugin_list);
+       GSList *plugins, *cur, *unloaded;
+       const gchar *text;
+       GtkListStore *store;
+       GtkTreeIter iter;
        GtkTextBuffer *textbuf;
        GtkTextIter start_iter, end_iter;
+       GtkTreeSelection *selection;
 
        plugins = plugin_get_list();
-       gtk_clist_freeze(clist);
-       gtk_clist_clear(clist);
+       unloaded = plugin_get_unloaded_list();
+
+       store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW
+                               (pluginwindow->plugin_list_view)));
+       gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(store),
+                                             0, GTK_SORT_ASCENDING);
+       gtk_list_store_clear(store);
+       
        textbuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(pluginwindow->plugin_desc));
+       gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(pluginwindow->plugin_desc), FALSE);
+       gtk_text_view_set_editable(GTK_TEXT_VIEW(pluginwindow->plugin_desc), FALSE);
        gtk_text_buffer_get_start_iter(textbuf, &start_iter);
        gtk_text_buffer_get_end_iter(textbuf, &end_iter);
        gtk_text_buffer_delete(textbuf, &start_iter, &end_iter);
@@ -72,41 +99,68 @@ static void set_plugin_list(PluginWindow *pluginwindow)
        for(cur = plugins; cur != NULL; cur = g_slist_next(cur)) {
                Plugin *plugin = (Plugin *) cur->data;
 
-               text[0] = (gchar *) plugin_get_name(plugin);
-               row = gtk_clist_append(clist, text);
-               gtk_clist_set_row_data(clist, row, plugin);
+               gtk_list_store_append(store, &iter);
+               text = plugin_get_name(plugin); 
+               gtk_list_store_set(store, &iter,
+                                  PLUGINWINDOW_NAME, text,
+                                  PLUGINWINDOW_DATA, plugin,
+                                  PLUGINWINDOW_STYLE, PANGO_STYLE_NORMAL,
+                                  -1);
+       }
+
+       for(cur = unloaded; cur != NULL; cur = g_slist_next(cur)) {
+               Plugin *plugin = (Plugin *) cur->data;
+
+               gtk_list_store_append(store, &iter);
+               text = plugin_get_name(plugin);
+               gtk_list_store_set(store, &iter,
+                                  PLUGINWINDOW_NAME, text,
+                                  PLUGINWINDOW_DATA, plugin,
+                                  PLUGINWINDOW_STYLE, PANGO_STYLE_ITALIC,
+                                  -1);
        }
-       gtk_clist_thaw(clist);
-       if (pluginwindow->selected_plugin == NULL)
-               gtk_clist_select_row (clist, 0, -1);
+
+       if (pluginwindow->selected_plugin == NULL) { 
+               selection = gtk_tree_view_get_selection(GTK_TREE_VIEW
+                               (pluginwindow->plugin_list_view));
+               gtk_tree_selection_unselect_all(selection);                             
+       }               
+       g_slist_free(plugins);
 }
 
-static void select_row_cb(GtkCList *clist, gint row, gint column,
-                         GdkEventButton *event, PluginWindow *pluginwindow)
+static void select_row_cb(Plugin *plugin, PluginWindow *pluginwindow)
 {
-       Plugin *plugin;
        GtkTextView *plugin_desc = GTK_TEXT_VIEW(pluginwindow->plugin_desc);
        GtkTextBuffer *textbuf = gtk_text_view_get_buffer(plugin_desc);
        GtkTextIter start_iter, end_iter;
-       const gchar *text;
+       gchar *text;
 
-       plugin = (Plugin *) gtk_clist_get_row_data(clist, row);
        pluginwindow->selected_plugin = plugin;
 
        if (pluginwindow->selected_plugin != NULL) {
+               const gchar *desc = plugin_get_desc(plugin);
+               const gchar *err = plugin_get_error(plugin);
                gtk_text_buffer_get_start_iter(textbuf, &start_iter);
                gtk_text_buffer_get_end_iter(textbuf, &end_iter);
                gtk_text_buffer_delete(textbuf, &start_iter, &end_iter);
-               text = plugin_get_desc(plugin);
+               
+               if (err == NULL)
+                       text = g_strconcat(desc, _("\n\nVersion: "),
+                                  plugin_get_version(plugin), "\n", NULL);
+               else
+                       text = g_strconcat(_("Error: "),
+                                  err, "\n", _("Plugin is not functional."), 
+                                  "\n\n", desc, _("\n\nVersion: "),
+                                  plugin_get_version(plugin), "\n", NULL);
                gtk_text_buffer_insert(textbuf, &start_iter, text, strlen(text));
+               g_free(text);
                gtk_widget_set_sensitive(pluginwindow->unload_btn, TRUE);
        } else {
                gtk_widget_set_sensitive(pluginwindow->unload_btn, FALSE);
        }
 }
 
-static void unselect_row_cb(GtkCList * clist, gint row, gint column,
-                           GdkEventButton * event, PluginWindow *pluginwindow)
+static void unselect_row_cb(Plugin *plugin, PluginWindow *pluginwindow)
 {
        gtk_widget_set_sensitive(pluginwindow->unload_btn, FALSE);      
 }
@@ -123,19 +177,35 @@ static void unload_cb(GtkButton *button, PluginWindow *pluginwindow)
 
 static void load_cb(GtkButton *button, PluginWindow *pluginwindow)
 {
-       gchar *file, *error = NULL;
-
-       file = filesel_select_file_open(_("Select Plugin to load"), PLUGINDIR);
-       if (file == NULL)
-               return;
-
-       plugin_load(file, &error);
-       if (error != NULL) {
-               alertpanel_error("The following error occured while loading the plugin:\n%s\n", error);
-               g_free(error);
-       }
+       GList *file_list;
+
+       file_list = filesel_select_multiple_files_open_with_filter(
+                       _("Select Plugin to load"), get_plugin_dir(), 
+                       "*." G_MODULE_SUFFIX);
+
+       if (file_list) {
+               GList *tmp;
+
+               for ( tmp = file_list; tmp; tmp = tmp->next) {
+                       gchar *file, *error = NULL;
+
+                       file = (gchar *) tmp->data;
+                       if (!file) continue;
+                       plugin_load(file, &error);
+                       if (error != NULL) {
+                               alertpanel_error(
+                               _("The following error occured while loading the plugin [%s] :\n%s\n"),
+                               file, error);
+                               g_free(error);
+                       }
+
+                       /* FIXME: globally or atom-ly : ? */
+                       set_plugin_list(pluginwindow);
+                       g_free(file);
+               }
 
-       set_plugin_list(pluginwindow);          
+               g_list_free(file_list);
+       }               
 }
 
 static gboolean pluginwindow_key_pressed(GtkWidget *widget, GdkEventKey *event,
@@ -167,6 +237,19 @@ static gboolean pluginwindow_key_pressed(GtkWidget *widget, GdkEventKey *event,
        return FALSE;
 }
 
+/*!
+ *\brief       Save Gtk object size to prefs dataset
+ */
+static void pluginwindow_size_allocate_cb(GtkWidget *widget,
+                                        GtkAllocation *allocation)
+{
+       g_return_if_fail(allocation != NULL);
+
+       prefs_common.pluginswin_width = allocation->width;
+       prefs_common.pluginswin_height = allocation->height;
+}
+
+
 void pluginwindow_create()
 {
        PluginWindow *pluginwindow;
@@ -175,8 +258,7 @@ void pluginwindow_create()
        GtkWidget *vbox1;
        GtkWidget *hbox2;
        GtkWidget *scrolledwindow2;
-       GtkWidget *plugin_list;
-       GtkWidget *label12;
+       GtkWidget *plugin_list_view;
        GtkWidget *vbox2;
        GtkWidget *frame2;
        GtkWidget *label13;
@@ -186,10 +268,14 @@ void pluginwindow_create()
        GtkWidget *load_btn;
        GtkWidget *unload_btn;
        GtkWidget *close_btn;
+       static GdkGeometry geometry;
+       
+       debug_print("Creating plugins window...\n");
+
+       pluginwindow = g_new0(PluginWindow, 1);
 
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
        gtk_container_set_border_width(GTK_CONTAINER(window), 8);
-       gtk_window_set_default_size(GTK_WINDOW(window), 480, 300);
        gtk_window_set_title(GTK_WINDOW(window), _("Plugins"));
        gtk_window_set_modal(GTK_WINDOW(window), TRUE);
 
@@ -203,25 +289,15 @@ void pluginwindow_create()
 
        scrolledwindow2 = gtk_scrolled_window_new(NULL, NULL);
        gtk_widget_show(scrolledwindow2);
-       gtk_box_pack_start(GTK_BOX(hbox2), scrolledwindow2, TRUE, TRUE, 0);
+       gtk_box_pack_start(GTK_BOX(hbox2), scrolledwindow2, FALSE, FALSE, 0);
        gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW
                                       (scrolledwindow2), GTK_POLICY_NEVER,
-                                      GTK_POLICY_ALWAYS);
+                                      GTK_POLICY_AUTOMATIC);
 
-       plugin_list = gtk_clist_new(1);
-       gtk_widget_show(plugin_list);
-       gtk_container_add(GTK_CONTAINER(scrolledwindow2), plugin_list);
-       gtk_clist_set_column_width(GTK_CLIST(plugin_list), 0, 80);
-       gtk_clist_set_column_auto_resize(GTK_CLIST(plugin_list), 0, TRUE);
-       gtk_clist_column_titles_show(GTK_CLIST(plugin_list));
-       gtk_clist_set_selection_mode(GTK_CLIST (plugin_list), GTK_SELECTION_BROWSE);
-       gtk_widget_grab_focus(GTK_WIDGET(plugin_list));
-
-       label12 = gtk_label_new(_("Plugins"));
-       gtk_widget_show(label12);
-       gtk_clist_set_column_widget(GTK_CLIST(plugin_list), 0, label12);
-       gtk_label_set_justify(GTK_LABEL(label12), GTK_JUSTIFY_LEFT);
-       gtk_misc_set_alignment(GTK_MISC(label12), 0, 0.5);
+       plugin_list_view = pluginwindow_list_view_create(pluginwindow);
+       gtk_widget_show(plugin_list_view);
+       gtk_container_add(GTK_CONTAINER(scrolledwindow2), plugin_list_view);
+       gtk_widget_grab_focus(GTK_WIDGET(plugin_list_view));
 
        vbox2 = gtk_vbox_new(FALSE, 0);
        gtk_widget_show(vbox2);
@@ -253,9 +329,9 @@ void pluginwindow_create()
        gtk_box_pack_start(GTK_BOX(vbox1), hbuttonbox1, FALSE, FALSE, 0);
        gtk_button_box_set_layout(GTK_BUTTON_BOX(hbuttonbox1),
                                  GTK_BUTTONBOX_END);
-       gtk_button_box_set_spacing(GTK_BUTTON_BOX(hbuttonbox1), 0);
+       gtk_button_box_set_spacing(GTK_BUTTON_BOX(hbuttonbox1), 6);
 
-       load_btn = gtk_button_new_with_label(_("Load Plugin"));
+       load_btn = gtk_button_new_with_label(_("Load Plugin..."));
        gtk_widget_show(load_btn);
        gtk_container_add(GTK_CONTAINER(hbuttonbox1), load_btn);
        GTK_WIDGET_SET_FLAGS(load_btn, GTK_CAN_DEFAULT);
@@ -265,7 +341,7 @@ void pluginwindow_create()
        gtk_container_add(GTK_CONTAINER(hbuttonbox1), unload_btn);
        GTK_WIDGET_SET_FLAGS(unload_btn, GTK_CAN_DEFAULT);
 
-       close_btn = gtk_button_new_with_label(_("Close"));
+       close_btn = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
        gtk_widget_show(close_btn);
        gtk_container_add(GTK_CONTAINER(hbuttonbox1), close_btn);
        GTK_WIDGET_SET_FLAGS(close_btn, GTK_CAN_DEFAULT);
@@ -274,7 +350,6 @@ void pluginwindow_create()
        gtk_text_view_set_wrap_mode(GTK_TEXT_VIEW(plugin_desc), GTK_WRAP_WORD);
        gtk_widget_set_sensitive(GTK_WIDGET(unload_btn), FALSE);
 
-       pluginwindow = g_new0(PluginWindow, 1);
 
        g_signal_connect(G_OBJECT(load_btn), "released",
                         G_CALLBACK(load_cb), pluginwindow);
@@ -282,15 +357,13 @@ void pluginwindow_create()
                         G_CALLBACK(unload_cb), pluginwindow);
        g_signal_connect(G_OBJECT(close_btn), "released",
                         G_CALLBACK(close_cb), pluginwindow);
-       g_signal_connect(G_OBJECT(plugin_list), "select-row",
-                        G_CALLBACK(select_row_cb), pluginwindow);
-       g_signal_connect(G_OBJECT(plugin_list), "unselect-row",
-                        G_CALLBACK(unselect_row_cb), pluginwindow);
+       g_signal_connect(G_OBJECT(window), "size_allocate",
+                        G_CALLBACK(pluginwindow_size_allocate_cb), NULL);
        g_signal_connect(G_OBJECT(window), "key_press_event",
                           G_CALLBACK(pluginwindow_key_pressed), pluginwindow);
 
        pluginwindow->window = window;
-       pluginwindow->plugin_list = plugin_list;
+       pluginwindow->plugin_list_view = plugin_list_view;
        pluginwindow->plugin_desc = plugin_desc;
        pluginwindow->unload_btn = unload_btn;
        pluginwindow->selected_plugin = NULL;
@@ -298,5 +371,89 @@ void pluginwindow_create()
        set_plugin_list(pluginwindow);
 
        inc_lock();
+
+       if (!geometry.min_height) {
+               geometry.min_width = 480;
+               geometry.min_height = 300;
+       }
+
+       gtk_window_set_geometry_hints(GTK_WINDOW(window), NULL, &geometry,
+                                     GDK_HINT_MIN_SIZE);
+       gtk_widget_set_size_request(window, prefs_common.pluginswin_width,
+                                   prefs_common.pluginswin_height);
+
        gtk_widget_show(window);
 }
+
+static GtkListStore* pluginwindow_create_data_store(void)
+{
+       return gtk_list_store_new(N_PLUGINWINDOW_COLUMNS,
+                                 G_TYPE_STRING,        
+                                 G_TYPE_POINTER,
+                                 PANGO_TYPE_STYLE,
+                                 -1);
+}
+
+static GtkWidget *pluginwindow_list_view_create(PluginWindow *pluginwindow)
+{
+       GtkTreeView *list_view;
+       GtkTreeSelection *selector;
+       GtkTreeModel *model;
+
+       model = GTK_TREE_MODEL(pluginwindow_create_data_store());
+       list_view = GTK_TREE_VIEW(gtk_tree_view_new_with_model(model));
+       g_object_unref(model);  
+
+       gtk_tree_view_set_rules_hint(list_view, prefs_common.enable_rules_hint);
+       gtk_tree_view_set_search_column (list_view, 0);
+
+       selector = gtk_tree_view_get_selection(list_view);
+       gtk_tree_selection_set_mode(selector, GTK_SELECTION_BROWSE);
+       gtk_tree_selection_set_select_function(selector, pluginwindow_selected,
+                                              pluginwindow, NULL);
+
+       /* create the columns */
+       pluginwindow_create_list_view_columns(GTK_WIDGET(list_view));
+
+       return GTK_WIDGET(list_view);
+}
+
+static void pluginwindow_create_list_view_columns(GtkWidget *list_view)
+{
+       GtkTreeViewColumn *column;
+       GtkCellRenderer *renderer;
+
+       renderer = gtk_cell_renderer_text_new();
+       column = gtk_tree_view_column_new_with_attributes
+               (_("Plugins"),
+                renderer,
+                "text", PLUGINWINDOW_NAME,
+                "style", PLUGINWINDOW_STYLE,
+                NULL);
+       gtk_tree_view_append_column(GTK_TREE_VIEW(list_view), column);          
+}
+
+static gboolean pluginwindow_selected(GtkTreeSelection *selector,
+                                     GtkTreeModel *model, 
+                                     GtkTreePath *path,
+                                     gboolean currently_selected,
+                                     gpointer data)
+{
+       GtkTreeIter iter;
+       Plugin *plugin;
+       
+       if (!gtk_tree_model_get_iter(model, &iter, path))
+               return TRUE;
+
+       gtk_tree_model_get(model, &iter, 
+                          PLUGINWINDOW_DATA, &plugin,
+                          -1);
+
+       if (currently_selected) 
+               unselect_row_cb(plugin, data);
+       else
+               select_row_cb(plugin, data);
+
+       return TRUE;
+}
+