fix bug 4239, 'Preferences: Text Options Header Display modal is not modal' (sic)
[claws.git] / src / addrgather.c
index 032bbce9100230dff45ac6831aa91f8c4ecf0941..52313d73cc3fbb7162cc8d8da702978cd42d773d 100644 (file)
@@ -1,6 +1,6 @@
 /*
- * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
- * Copyright (C) 2002-2008 Match Grun and the Claws Mail team
+ * Claws Mail -- a GTK+ based, lightweight, and fast e-mail client
+ * Copyright (C) 2002-2018 Match Grun and the Claws Mail 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
@@ -14,7 +14,6 @@
  *
  * You should have received a copy of the GNU General Public License
  * along with this program. If not, see <http://www.gnu.org/licenses/>.
- * 
  */
 
 /*
@@ -23,6 +22,7 @@
 
 #ifdef HAVE_CONFIG_H
 #  include "config.h"
+#include "claws-features.h"
 #endif
 
 #include "defs.h"
 #include "manage_window.h"
 #include "folder.h"
 #include "utils.h"
+#include "prefs_common.h"
+#include "prefs_gtk.h"
 
 #include "addrharvest.h"
-#include "addrindex.h"
-#include "addrbook.h"
-
+#ifndef USE_ALT_ADDRBOOK
+       #include "addrindex.h"
+       #include "addrbook.h"
+#else
+       #include "addressbook-dbus.h"
+#endif
 #define PAGE_FIELDS     0
 #define PAGE_FINISH     1
 
@@ -73,7 +78,9 @@ static struct _AddrHarvest {
        GtkWidget *labelFolder;
        GtkWidget *entryBook;
        GtkWidget *checkHeader[ NUM_FIELDS ];
+#ifndef USE_ALT_ADDRBOOK
        GtkWidget *spinbtnFolder;
+#endif
        GtkWidget *checkRecurse;
        GtkWidget *btnOk;
        GtkWidget *btnCancel;
@@ -82,11 +89,13 @@ static struct _AddrHarvest {
        gboolean  cancelled;
        gboolean  done;
        gchar     *folderPath;
-       GtkWidget *clistCount;
+       GtkWidget *viewCount;
 } addrgather_dlg;
 
+#ifndef USE_ALT_ADDRBOOK
 static AddressIndex *_harv_addressIndex_;
 static AddressBookFile *_harv_addressBook_;
+#endif
 static gchar *_harv_headerNames_[] = {
        HEADER_FROM,
        HEADER_REPLY_TO,
@@ -97,6 +106,12 @@ static gchar *_harv_headerNames_[] = {
 };
 static GList *_harv_messageList_;
 
+enum {
+       ADDRGATHER_COL_HEADER,
+       ADDRGATHER_COL_COUNT,
+       N_ADDRGATHER_COLS
+};
+
 static void addrgather_dlg_status_show( gchar *msg ) {
        if( addrgather_dlg.statusbar != NULL ) {
                gtk_statusbar_pop( GTK_STATUSBAR(addrgather_dlg.statusbar),
@@ -120,25 +135,42 @@ static gint addrgather_dlg_delete_event(
 static gboolean addrgather_dlg_key_pressed(
        GtkWidget *widget, GdkEventKey *event, gpointer data )
 {
-       if( event && event->keyval == GDK_Escape ) {
+       if( event && event->keyval == GDK_KEY_Escape ) {
                addrgather_dlg.cancelled = TRUE;
                gtk_main_quit();
        }
        return FALSE;
 }
 
+static void addrgather_size_allocate(
+       GtkWidget *widget, GtkAllocation *allocation )
+{
+       cm_return_if_fail( allocation != NULL );
+       
+       prefs_common.addrgather_width   = allocation->width;
+       prefs_common.addrgather_height  = allocation->height;
+}
+
 #define FMT_BUFSIZE 32
 
 static gboolean addrgather_dlg_harvest() {
-       GtkCMCList *clist;
-       gchar *text[ FIELDS_N_COLS ];
+       GtkWidget *view;
+       GtkTreeModel *model;
+       GtkTreeIter iter;
        AddressHarvester *harvester;
-       AddressBookFile *abf;
        gchar *name;
+#ifndef USE_ALT_ADDRBOOK
+       AddressBookFile *abf;
        gchar *newFile;
+#else
+    GList* list;
+    ContactData* contact;
+    GError* error = NULL;
+#endif
        gchar str[ FMT_BUFSIZE ];
        gint cnt;
        gint i;
+#ifndef USE_ALT_ADDRBOOK
        gint sz;
 
        name = gtk_editable_get_chars( GTK_EDITABLE(addrgather_dlg.entryBook), 0, -1 );
@@ -147,6 +179,14 @@ static gboolean addrgather_dlg_harvest() {
                        _( "Please specify name for address book." ) );
                g_free( name );
                return FALSE;
+#else
+       name = gtk_editable_get_chars( GTK_EDITABLE(addrgather_dlg.entryBook), 0, -1 );
+       if( name == NULL || strlen( name ) < 1 ) {
+               addrgather_dlg_status_show(
+                       _("No available address book."));
+               g_free( name );
+               return FALSE;
+#endif
        }
 
        /* Create harvest helper */
@@ -173,10 +213,13 @@ static gboolean addrgather_dlg_harvest() {
        /* Go fer it */
        addrgather_dlg_status_show( _( "Collecting addresses..." ) );
        GTK_EVENTS_FLUSH();
+#ifndef USE_ALT_ADDRBOOK
        sz = gtk_spin_button_get_value_as_int(
                GTK_SPIN_BUTTON( addrgather_dlg.spinbtnFolder ) );
        addrharvest_set_folder_size( harvester, sz );
+#endif
 
+#ifndef USE_ALT_ADDRBOOK
        /* Create address book */
        abf = addrbook_create_book();
        addrbook_set_path( abf, _harv_addressIndex_->filePath );
@@ -185,27 +228,56 @@ static gboolean addrgather_dlg_harvest() {
        addrbook_set_name( abf, name );
        g_free( newFile );
        g_free( name );
+#endif
 
        /* Harvest addresses */
+#ifndef USE_ALT_ADDRBOOK
        addrharvest_harvest(
                harvester, abf->addressCache, _harv_messageList_ );
+       /* save address book */
        addrbook_save_data( abf );
        _harv_addressBook_ = abf;
+#else
+       addrharvest_harvest(
+               harvester, NULL, _harv_messageList_ );
+       list = g_hash_table_get_values(harvester->dupTable);
+       for (; list; list = g_list_next(list)) {
+               contact = g_new0(ContactData, 1);
+               ContactEntry* person = (ContactEntry *) list->data;
+               if (person->first_name)
+                       contact->name = g_strconcat(
+                                       person->first_name, " ", person->last_name, NULL);
+               else
+                       contact->name = g_strdup(person->last_name);
+               if (! contact->name || strlen(contact->name) < 1)
+                       contact->name = g_strdup(person->email);
+               contact->remarks = g_strdup(N_("address added by Claws Mail"));
+               contact->email = g_strdup(person->email);
+               contact->book = g_strdup(name);
+               addressbook_dbus_add_contact(contact, &error);
+               contact_data_free(&contact);
+       }
+       g_list_free(list);
+       g_free(name);
+#endif
 
        /* Update summary count */
-       clist = GTK_CMCLIST(addrgather_dlg.clistCount);
-       gtk_cmclist_clear( clist );
+       view = addrgather_dlg.viewCount;
+       model = gtk_tree_view_get_model(GTK_TREE_VIEW(view));
+       gtk_list_store_clear( GTK_LIST_STORE(model) );
        for( i = 0; i < NUM_FIELDS; i++ ) {
                cnt = addrharvest_get_count( harvester, _harv_headerNames_[i] );
                if( cnt < 1 ) {
                        strcpy( str, "-" );
                }
                else {
-                       sprintf( str, "%d", cnt );
+                       snprintf( str, FMT_BUFSIZE, "%d", cnt );
                }
-               text[ FIELD_COL_HEADER ] = _harv_headerNames_[i];
-               text[ FIELD_COL_COUNT  ] = str;
-               gtk_cmclist_append( clist, text );
+               gtk_list_store_append(GTK_LIST_STORE(model), &iter);
+               gtk_list_store_set(GTK_LIST_STORE(model), &iter,
+                               ADDRGATHER_COL_HEADER, _harv_headerNames_[i],
+                               ADDRGATHER_COL_COUNT, str,
+                               -1);
        }
 
        addrharvest_free( harvester );
@@ -237,13 +309,6 @@ static void addrgather_dlg_cancel( GtkWidget *widget, gpointer data ) {
        gtk_main_quit();
 }
 
-#define PACK_CHECK_BUTTON(box, checkbtn, label) \
-{ \
-       checkbtn = gtk_check_button_new_with_label(label); \
-       gtk_widget_show(checkbtn); \
-       gtk_box_pack_start(GTK_BOX(box), checkbtn, FALSE, TRUE, 0); \
-}
-
 /*
  * Create notebook page for mail headers.
  * Enter: pageNum Page number.
@@ -253,19 +318,24 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
 {
        GtkWidget *vbox;
        GtkWidget *vboxf;
-       GtkWidget *hboxs;
        GtkWidget *table;
        GtkWidget *label;
        GtkWidget *labelFolder;
        GtkWidget *entryBook;
        GtkWidget *frameHeader;
        GtkWidget *checkHeader[NUM_FIELDS];
+#ifndef USE_ALT_ADDRBOOK
+       GtkWidget *hboxs;
        GtkWidget *spinbtnFolder;
-       GtkObject *adjFolder;
+       GtkAdjustment *adjFolder;
+#endif
        GtkWidget *checkRecurse;
        gint top;
        gint i;
-       CLAWS_TIP_DECL();
+#ifdef USE_ALT_ADDRBOOK
+       GError* error = NULL;
+       GSList *books, *cur;
+#endif
 
        /* Container */
        vbox = gtk_vbox_new(FALSE, 6);
@@ -302,10 +372,27 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
        gtk_table_attach( GTK_TABLE(table), label, 0, 1, top, (top + 1), GTK_FILL, 0, 0, 0 );
        gtk_misc_set_alignment( GTK_MISC(label), 1.0, 0.5 );
 
+#ifndef USE_ALT_ADDRBOOK
        entryBook = gtk_entry_new();
+#else
+        books = addressbook_dbus_get_books(&error);
+        entryBook = gtk_combo_box_text_new();
+        if (books) {
+            for (cur = books; cur; cur = g_slist_next(cur)) {
+                gchar* book = (gchar *) cur->data;
+                gtk_combo_box_text_prepend_text(GTK_COMBO_BOX_TEXT(entryBook), book);
+                g_free(book);
+            }
+            g_slist_free(books);
+               }
+        else
+            gtk_combo_box_text_prepend_text(GTK_COMBO_BOX_TEXT(entryBook), "");
+        gtk_combo_box_set_active(GTK_COMBO_BOX(entryBook), 0);
+#endif
        gtk_table_attach( GTK_TABLE(table), entryBook, 1, 2, top, (top + 1),
                GTK_EXPAND|GTK_SHRINK|GTK_FILL, 0, 0, 0 );
 
+#ifndef USE_ALT_ADDRBOOK
        /* Third row */
        top = 2;
        label = gtk_label_new(_("Address book folder size:"));
@@ -315,15 +402,14 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
                        _("Maximum amount of entries per folder within the newly created address book"));
 
        hboxs = gtk_hbox_new(FALSE, 8);
-       adjFolder = gtk_adjustment_new(DFL_FOLDER_SIZE, MIN_FOLDER_SIZE, G_MAXINT, 1, 10, 0);
+       adjFolder = GTK_ADJUSTMENT(gtk_adjustment_new(DFL_FOLDER_SIZE, MIN_FOLDER_SIZE, G_MAXINT, 1, 10, 0));
        spinbtnFolder = gtk_spin_button_new(GTK_ADJUSTMENT(adjFolder), 1, 0);
        gtk_box_pack_start(GTK_BOX(hboxs), spinbtnFolder, FALSE, FALSE, 0);
-       gtk_widget_set_size_request(spinbtnFolder, 100, -1);
        gtk_spin_button_set_numeric(GTK_SPIN_BUTTON(spinbtnFolder), TRUE);
        gtk_table_attach(GTK_TABLE(table), hboxs, 1, 2, top, (top + 1), GTK_FILL, 0, 0, 0);
        CLAWS_SET_TIP(spinbtnFolder,
                        _("Maximum amount of entries per folder within the newly created address book"));
-
+#endif
        /* Fourth row */
        top = 3;
        frameHeader = gtk_frame_new(_("Process these mail header fields"));
@@ -338,7 +424,8 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
        gtk_container_set_border_width(GTK_CONTAINER(vboxf), 8);
 
        for (i = 0; i < NUM_FIELDS; i++) {
-               PACK_CHECK_BUTTON(vboxf, checkHeader[i], _harv_headerNames_[i]);
+               PACK_CHECK_BUTTON(vboxf, checkHeader[i],
+                       prefs_common_translated_header_name(_harv_headerNames_[i]));
                addrgather_dlg.checkHeader[i] = checkHeader[i];
        }
 
@@ -350,7 +437,9 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
 
        addrgather_dlg.labelFolder   = labelFolder;
        addrgather_dlg.entryBook     = entryBook;
+#ifndef USE_ALT_ADDRBOOK
        addrgather_dlg.spinbtnFolder = spinbtnFolder;
+#endif
        addrgather_dlg.checkRecurse  = checkRecurse;
 }
 
@@ -362,13 +451,12 @@ static void addrgather_page_fields(gint pageNum, gchar *pageLbl)
 static void addrgather_page_finish( gint pageNum, gchar *pageLbl ) {
        GtkWidget *label;
        GtkWidget *vbox;
-       GtkWidget *clistSWin;
-       GtkWidget *clistCount;
-       gchar *titles[ FIELDS_N_COLS ];
-       gint i;
-
-       titles[ FIELD_COL_HEADER ] = _("Header Name");
-       titles[ FIELD_COL_COUNT  ] = _("Address Count");
+       GtkWidget *scrollwin;
+       GtkWidget *viewCount;
+       GtkTreeViewColumn *col;
+       GtkCellRenderer *rdr;
+       GtkTreeSelection *sel;
+       GtkTreeModel *model;
 
        vbox = gtk_vbox_new(FALSE, 8);
        gtk_container_add( GTK_CONTAINER( addrgather_dlg.notebook ), vbox );
@@ -382,24 +470,40 @@ static void addrgather_page_finish( gint pageNum, gchar *pageLbl ) {
                label );
 
        /* Summary count */
-       clistSWin = gtk_scrolled_window_new( NULL, NULL );
-       gtk_container_add( GTK_CONTAINER(vbox), clistSWin );
-       gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(clistSWin),
+       scrollwin = gtk_scrolled_window_new( NULL, NULL );
+       gtk_container_add( GTK_CONTAINER(vbox), scrollwin );
+       gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollwin),
                                       GTK_POLICY_AUTOMATIC,
                                       GTK_POLICY_AUTOMATIC);
 
-       clistCount = gtk_cmclist_new_with_titles( FIELDS_N_COLS, titles );
-       gtk_container_add( GTK_CONTAINER(clistSWin), clistCount );
-       gtk_cmclist_set_selection_mode( GTK_CMCLIST(clistCount), GTK_SELECTION_BROWSE );
-       gtk_cmclist_set_column_width(
-                       GTK_CMCLIST(clistCount), FIELD_COL_HEADER, FIELDS_COL_WIDTH_HEADER );
-       gtk_cmclist_set_column_width(
-                       GTK_CMCLIST(clistCount), FIELD_COL_COUNT, FIELDS_COL_WIDTH_COUNT );
-
-       for( i = 0; i < FIELDS_N_COLS; i++ )
-               GTK_WIDGET_UNSET_FLAGS(GTK_CMCLIST(clistCount)->column[i].button, GTK_CAN_FOCUS);
-
-       addrgather_dlg.clistCount = clistCount;
+       /* Treeview */
+       model = GTK_TREE_MODEL(gtk_list_store_new(N_ADDRGATHER_COLS,
+                       G_TYPE_STRING, G_TYPE_STRING, -1));
+
+       viewCount = gtk_tree_view_new_with_model(GTK_TREE_MODEL(model));
+       g_object_unref(model);
+       gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(viewCount), TRUE);
+       gtk_tree_view_set_reorderable(GTK_TREE_VIEW(viewCount), FALSE);
+       sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(viewCount));
+       gtk_tree_selection_set_mode(sel, GTK_SELECTION_NONE);
+
+       /* Columns for the treeview */
+       rdr = gtk_cell_renderer_text_new();
+       col = gtk_tree_view_column_new_with_attributes(_("Header Name"), rdr,
+                       "markup", ADDRGATHER_COL_HEADER,
+                       NULL);
+       gtk_tree_view_column_set_min_width(col, FIELDS_COL_WIDTH_HEADER);
+       gtk_tree_view_append_column(GTK_TREE_VIEW(viewCount), col);
+
+       col = gtk_tree_view_column_new_with_attributes(_("Address Count"), rdr,
+                       "text", ADDRGATHER_COL_COUNT,
+                       NULL);
+       gtk_tree_view_column_set_min_width(col, FIELDS_COL_WIDTH_COUNT);
+       gtk_tree_view_append_column(GTK_TREE_VIEW(viewCount), col);
+
+       gtk_container_add( GTK_CONTAINER(scrollwin), viewCount );
+
+       addrgather_dlg.viewCount = viewCount;
 }
 
 /*
@@ -417,18 +521,20 @@ static void addrgather_dlg_create(void)
        GtkWidget *vbox;
        GtkWidget *hbbox;
        GtkWidget *hsbox;
-
+       static GdkGeometry geometry;
+       
        window = gtkut_window_new(GTK_WINDOW_TOPLEVEL, "addrgather");
-       gtk_widget_set_size_request(window, 380, -1);
        gtk_container_set_border_width(GTK_CONTAINER(window), 4);
        gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
        gtk_window_set_resizable(GTK_WINDOW(window), TRUE);
-       gtk_window_set_modal(GTK_WINDOW(window), TRUE);
+       gtk_window_set_type_hint(GTK_WINDOW(window), GDK_WINDOW_TYPE_HINT_DIALOG);
        
        g_signal_connect(G_OBJECT(window), "delete_event",
                         G_CALLBACK(addrgather_dlg_delete_event), NULL);
        g_signal_connect(G_OBJECT(window), "key_press_event",
                         G_CALLBACK(addrgather_dlg_key_pressed), NULL);
+       g_signal_connect(G_OBJECT(window), "size_allocate",
+                        G_CALLBACK(addrgather_size_allocate), NULL);
 
        vbox = gtk_vbox_new(FALSE, 6);
        gtk_container_add(GTK_CONTAINER(window), vbox);
@@ -458,7 +564,15 @@ static void addrgather_dlg_create(void)
        g_signal_connect(G_OBJECT(btnCancel), "clicked",
                         G_CALLBACK(addrgather_dlg_cancel), NULL);
 
-       gtk_widget_show_all(vbox);
+       if (!geometry.min_width) {
+               geometry.min_width = 450;
+               geometry.min_height = -1;
+       }
+
+       gtk_window_set_geometry_hints(GTK_WINDOW(window), NULL, &geometry,
+                                     GDK_HINT_MIN_SIZE);
+       gtk_window_set_default_size(GTK_WINDOW(window), prefs_common.addrgather_width,
+                                   prefs_common.addrgather_height);
 
        addrgather_dlg.window     = window;
        addrgather_dlg.notebook   = notebook;
@@ -482,13 +596,19 @@ static void addrgather_dlg_create(void)
  *        msgList    List of message numbers, or NULL to process folder.
  * Return: Populated address book file, or NULL if none created.
  */
+#ifndef USE_ALT_ADDRBOOK
 AddressBookFile *addrgather_dlg_execute(FolderItem *folderItem, AddressIndex *addrIndex,
                                        gboolean sourceInd, GList *msgList)
+#else
+void addrgather_dlg_execute(FolderItem *folderItem, gboolean sourceInd, GList *msgList)
+#endif
 {
        gint i;
 
+#ifndef USE_ALT_ADDRBOOK
        _harv_addressIndex_ = addrIndex;
        _harv_addressBook_ = NULL;
+#endif
        _harv_messageList_ = msgList;
 
        /* Create dialog */
@@ -501,8 +621,11 @@ AddressBookFile *addrgather_dlg_execute(FolderItem *folderItem, AddressIndex *ad
        addrgather_dlg.folderPath = folder_item_get_path(folderItem);
 
        /* Setup some default values */
+
        gtk_label_set_text(GTK_LABEL(addrgather_dlg.labelFolder), folderItem->path);
+#ifndef USE_ALT_ADDRBOOK
        gtk_entry_set_text(GTK_ENTRY(addrgather_dlg.entryBook), folderItem->path);
+#endif
 
        for (i = 0; i < NUM_FIELDS; i++) {
                gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(addrgather_dlg.checkHeader[i]),
@@ -513,6 +636,7 @@ AddressBookFile *addrgather_dlg_execute(FolderItem *folderItem, AddressIndex *ad
        }
 
        gtk_widget_set_sensitive(addrgather_dlg.btnOk, TRUE);
+       gtk_widget_set_sensitive(addrgather_dlg.btnCancel, TRUE);
        gtk_widget_grab_default(addrgather_dlg.btnOk);
 
        /* Apply window title */
@@ -529,7 +653,7 @@ AddressBookFile *addrgather_dlg_execute(FolderItem *folderItem, AddressIndex *ad
 
        addrgather_dlg_status_show("");
        gtk_widget_show(addrgather_dlg.window);
-
+       gtk_window_set_modal(GTK_WINDOW(addrgather_dlg.window), TRUE);
        gtk_widget_grab_focus(addrgather_dlg.entryBook);
        manage_window_set_transient(GTK_WINDOW(addrgather_dlg.window));
        gtk_main();
@@ -537,14 +661,16 @@ AddressBookFile *addrgather_dlg_execute(FolderItem *folderItem, AddressIndex *ad
        g_free(addrgather_dlg.folderPath);
        addrgather_dlg.folderPath = NULL;
        gtk_widget_hide(addrgather_dlg.window);
+       gtk_window_set_modal(GTK_WINDOW(addrgather_dlg.window), FALSE);
+#ifndef USE_ALT_ADDRBOOK
        _harv_addressIndex_ = NULL;
 
        if (addrgather_dlg.cancelled == TRUE)
                return NULL;
 
        return _harv_addressBook_;
+#endif
 }
-
 /*
 * End of Source.
 */