diff options
-rw-r--r-- | addressbook/ChangeLog | 28 | ||||
-rw-r--r-- | addressbook/gui/component/addressbook.c | 129 | ||||
-rw-r--r-- | addressbook/gui/component/addressbook.h | 1 | ||||
-rw-r--r-- | addressbook/gui/contact-editor/e-contact-editor.c | 191 | ||||
-rw-r--r-- | addressbook/gui/contact-editor/e-contact-editor.h | 3 |
5 files changed, 259 insertions, 93 deletions
diff --git a/addressbook/ChangeLog b/addressbook/ChangeLog index 8b6e1b0f79..4f2d70de5c 100644 --- a/addressbook/ChangeLog +++ b/addressbook/ChangeLog @@ -1,5 +1,33 @@ 2003-12-16 Hans Petter Jansson <hpj@ximian.com> + * gui/component/addressbook.c (addressbook_show_load_error_dialog): + Make a public function that shows an error dialog, which can be used + when a source fails to load. + (book_open_cb): Call addressbook_show_load_error_dialog () on error. + (set_prop): Don't reuse the book when loading a new source. This was + confusing the contact-editor, which gets the book passed in. + + * gui/contact-editor/e-contact-editor.c (e_contact_editor_class_init): + Make "source_book" and "target_book" distinct properties. + (new_target_cb): Implement. + (source_selected): Implement. + (contact_moved_cb): Implement. + (contact_added_cb): Handle the case where source_book != target_book as + a move - if the add was successful, remove contact from source book. + (save_contact): Handle case where source_book != target_book. + (delete_cb): "book" renamed to "source_book". + (e_contact_editor_init): Set up source_selected signal. + (e_contact_editor_dispose): Dispose of both source and target books. + (e_contact_editor_new): "book" renamed to "source_book". + (e_contact_editor_set_property): If a source book is set, and we don't + have a target book, make target_book == source_book. Allow setting + target book, and base editability on that. If we're editing an existing + contact, treat book change as a change, and update command state. + (e_contact_editor_get_property): Allow getting target book. + (set_source_field): Indicate the selected book. + +2003-12-16 Hans Petter Jansson <hpj@ximian.com> + * gui/contact-editor/e-contact-editor.c (set_source_field): Implement. Makes the source menu reflect the source for the current book. diff --git a/addressbook/gui/component/addressbook.c b/addressbook/gui/component/addressbook.c index f415c903e9..7d67f447a2 100644 --- a/addressbook/gui/component/addressbook.c +++ b/addressbook/gui/component/addressbook.c @@ -473,6 +473,69 @@ addressbook_view_unref (AddressbookView *view) } } +void +addressbook_show_load_error_dialog (GtkWidget *parent, ESource *source, EBookStatus status) +{ + char *label_string; + GtkWidget *warning_dialog; + GtkWidget *href = NULL; + gchar *uri; + + g_return_if_fail (source != NULL); + + uri = e_source_get_uri (source); + + if (!strncmp (uri, "file:", 5)) { + label_string = + _("We were unable to open this addressbook. Please check that the\n" + "path exists and that you have permission to access it."); + } + else if (!strncmp (uri, "ldap:", 5)) { + /* special case for ldap: contact folders so we can tell the user about openldap */ +#if HAVE_LDAP + label_string = + _("We were unable to open this addressbook. This either\n" + "means you have entered an incorrect URI, or the LDAP server\n" + "is unreachable."); +#else + label_string = + _("This version of Evolution does not have LDAP support\n" + "compiled in to it. If you want to use LDAP in Evolution\n" + "you must compile the program from the CVS sources after\n" + "retrieving OpenLDAP from the link below.\n"); + href = gnome_href_new ("http://www.openldap.org/", "OpenLDAP at http://www.openldap.org/"); +#endif + } else { + /* other network folders */ + label_string = + _("We were unable to open this addressbook. This either\n" + "means you have entered an incorrect URI, or the server\n" + "is unreachable."); + } + + warning_dialog = gtk_message_dialog_new (parent ? GTK_WINDOW (parent) : NULL, + 0, + GTK_MESSAGE_WARNING, + GTK_BUTTONS_CLOSE, + label_string, + NULL); + + g_signal_connect (warning_dialog, + "response", + G_CALLBACK (gtk_widget_destroy), + warning_dialog); + + gtk_window_set_title (GTK_WINDOW (warning_dialog), _("Unable to open addressbook")); + + if (href) + gtk_box_pack_start (GTK_BOX (GTK_DIALOG (warning_dialog)->vbox), + href, FALSE, FALSE, 0); + + gtk_widget_show_all (warning_dialog); + + g_free (uri); +} + static void book_open_cb (EBook *book, EBookStatus status, gpointer closure) { @@ -486,65 +549,8 @@ book_open_cb (EBook *book, EBookStatus status, gpointer closure) view->book = book; } else { - char *label_string; - GtkWidget *warning_dialog; - GtkWidget *href = NULL; - gchar *uri; - view->failed_to_load = TRUE; - - uri = e_source_get_uri (view->source); - - if (!strncmp (uri, "file:", 5)) { - label_string = - _("We were unable to open this addressbook. Please check that the\n" - "path exists and that you have permission to access it."); - } - else if (!strncmp (uri, "ldap:", 5)) { - /* special case for ldap: contact folders so we can tell the user about openldap */ -#if HAVE_LDAP - label_string = - _("We were unable to open this addressbook. This either\n" - "means you have entered an incorrect URI, or the LDAP server\n" - "is unreachable."); -#else - label_string = - _("This version of Evolution does not have LDAP support\n" - "compiled in to it. If you want to use LDAP in Evolution\n" - "you must compile the program from the CVS sources after\n" - "retrieving OpenLDAP from the link below.\n"); - href = gnome_href_new ("http://www.openldap.org/", "OpenLDAP at http://www.openldap.org/"); -#endif - } else { - /* other network folders */ - label_string = - _("We were unable to open this addressbook. This either\n" - "means you have entered an incorrect URI, or the server\n" - "is unreachable."); - } - - warning_dialog = gtk_message_dialog_new ( - NULL /* XXX */, - 0, - GTK_MESSAGE_WARNING, - GTK_BUTTONS_CLOSE, - label_string, - NULL); - - g_signal_connect (warning_dialog, - "response", - G_CALLBACK (gtk_widget_destroy), - warning_dialog); - - gtk_window_set_title (GTK_WINDOW (warning_dialog), _("Unable to open addressbook")); - - if (href) - gtk_box_pack_start (GTK_BOX (GTK_DIALOG (warning_dialog)->vbox), - href, FALSE, FALSE, 0); - - gtk_widget_show_all (warning_dialog); - - g_free (uri); + addressbook_show_load_error_dialog (NULL /* XXX */, view->source, status); } } @@ -765,13 +771,12 @@ set_prop (BonoboPropertyBag *bag, case PROPERTY_SOURCE_UID_IDX: if (view->book) { - /* we've already had a uri set on this view, so unload it */ - e_book_async_unload_uri (view->book); + g_object_unref (view->book); view->source = NULL; - } else { - view->book = e_book_new (); } + view->book = e_book_new (); + view->failed_to_load = FALSE; uid = BONOBO_ARG_GET_STRING (arg); diff --git a/addressbook/gui/component/addressbook.h b/addressbook/gui/component/addressbook.h index fb05287133..16395789e2 100644 --- a/addressbook/gui/component/addressbook.h +++ b/addressbook/gui/component/addressbook.h @@ -14,6 +14,7 @@ void addressbook_load_uri (EBook *book, const char *uri, EBook #endif void addressbook_load_source (EBook *book, ESource *source, EBookCallback cb, gpointer closure); void addressbook_load_default_book (EBookCallback open_response, gpointer closure); +void addressbook_show_load_error_dialog (GtkWidget *parent, ESource *source, EBookStatus status); BonoboControl *addressbook_new_control (void); diff --git a/addressbook/gui/contact-editor/e-contact-editor.c b/addressbook/gui/contact-editor/e-contact-editor.c index 517d21b9bb..71ac2949d9 100644 --- a/addressbook/gui/contact-editor/e-contact-editor.c +++ b/addressbook/gui/contact-editor/e-contact-editor.c @@ -49,6 +49,7 @@ #include <e-util/e-categories-master-list-wombat.h> +#include "addressbook/gui/component/addressbook.h" #include "addressbook/printing/e-contact-print.h" #include "addressbook/printing/e-contact-print-envelope.h" #include "addressbook/gui/widgets/eab-gui-util.h" @@ -105,7 +106,8 @@ static guint contact_editor_signals[LAST_SIGNAL]; /* The arguments we take */ enum { PROP_0, - PROP_BOOK, + PROP_SOURCE_BOOK, + PROP_TARGET_BOOK, PROP_CONTACT, PROP_IS_NEW_CONTACT, PROP_EDITABLE, @@ -184,9 +186,16 @@ e_contact_editor_class_init (EContactEditorClass *klass) object_class->get_property = e_contact_editor_get_property; object_class->dispose = e_contact_editor_dispose; - g_object_class_install_property (object_class, PROP_BOOK, - g_param_spec_object ("book", - _("Book"), + g_object_class_install_property (object_class, PROP_SOURCE_BOOK, + g_param_spec_object ("source_book", + _("Source Book"), + /*_( */"XXX blurb" /*)*/, + E_TYPE_BOOK, + G_PARAM_READWRITE)); + + g_object_class_install_property (object_class, PROP_TARGET_BOOK, + g_param_spec_object ("target_book", + _("Target Book"), /*_( */"XXX blurb" /*)*/, E_TYPE_BOOK, G_PARAM_READWRITE)); @@ -503,6 +512,41 @@ address_mailing_changed (GtkWidget *widget, EContactEditor *editor) #endif } +static void +new_target_cb (EBook *new_book, EBookStatus status, EContactEditor *editor) +{ + if (status != E_BOOK_ERROR_OK || new_book == NULL) { + GtkWidget *source_option_menu; + + addressbook_show_load_error_dialog (NULL, e_book_get_source (new_book), status); + + source_option_menu = glade_xml_get_widget (editor->gui, "source-option-menu-source"); + e_source_option_menu_select (E_SOURCE_OPTION_MENU (source_option_menu), + e_book_get_source (editor->target_book)); + g_object_unref (new_book); + return; + } + + g_object_set (editor, "target_book", new_book, NULL); + g_object_unref (new_book); +} + +static void +source_selected (GtkWidget *source_option_menu, ESource *source, EContactEditor *editor) +{ + EBook *new_book; + + if (e_source_equal (e_book_get_source (editor->target_book), source)) + return; + + if (e_source_equal (e_book_get_source (editor->source_book), source)) { + g_object_set (editor, "target_book", editor->source_book, NULL); + return; + } + + new_book = e_book_new (); + addressbook_load_source (new_book, source, (EBookCallback) new_target_cb, editor); +} /* This function tells you whether name_to_style will make sense. */ static gboolean @@ -961,10 +1005,11 @@ categories_clicked(GtkWidget *button, EContactEditor *editor) typedef struct { EContactEditor *ce; gboolean should_close; + gchar *new_id; } EditorCloseStruct; static void -contact_added_cb (EBook *book, EBookStatus status, const char *id, EditorCloseStruct *ecs) +contact_moved_cb (EBook *book, EBookStatus status, EditorCloseStruct *ecs) { EContactEditor *ce = ecs->ce; gboolean should_close = ecs->should_close; @@ -972,7 +1017,47 @@ contact_added_cb (EBook *book, EBookStatus status, const char *id, EditorCloseSt gtk_widget_set_sensitive (ce->app, TRUE); ce->in_async_call = FALSE; - e_contact_set (ce->contact, E_CONTACT_UID, (char*)id); + e_contact_set (ce->contact, E_CONTACT_UID, ecs->new_id); + + g_signal_emit (ce, contact_editor_signals[CONTACT_DELETED], 0, + status, ce->contact); + + ce->is_new_contact = FALSE; + + if (should_close) { + close_dialog (ce); + } + else { + ce->changed = FALSE; + + g_object_ref (ce->target_book); + g_object_unref (ce->source_book); + ce->source_book = ce->target_book; + + command_state_changed (ce); + } + + g_object_unref (ce); + g_free (ecs->new_id); + g_free (ecs); +} + +static void +contact_added_cb (EBook *book, EBookStatus status, const char *id, EditorCloseStruct *ecs) +{ + EContactEditor *ce = ecs->ce; + gboolean should_close = ecs->should_close; + + if (ce->source_book != ce->target_book && status == E_BOOK_ERROR_OK && + ce->is_new_contact == FALSE) { + ecs->new_id = g_strdup (id); + e_book_async_remove_contact (ce->source_book, ce->contact, + (EBookCallback) contact_moved_cb, ecs); + return; + } + + gtk_widget_set_sensitive (ce->app, TRUE); + ce->in_async_call = FALSE; g_signal_emit (ce, contact_editor_signals[CONTACT_ADDED], 0, status, ce->contact); @@ -1023,23 +1108,31 @@ contact_modified_cb (EBook *book, EBookStatus status, EditorCloseStruct *ecs) static void save_contact (EContactEditor *ce, gboolean should_close) { + EditorCloseStruct *ecs = g_new(EditorCloseStruct, 1); + extract_info (ce); + if (!ce->target_book) + return; - if (ce->book) { - EditorCloseStruct *ecs = g_new(EditorCloseStruct, 1); - - ecs->ce = ce; - g_object_ref (ecs->ce); + ecs->ce = ce; + g_object_ref (ecs->ce); - ecs->should_close = should_close; + ecs->should_close = should_close; - gtk_widget_set_sensitive (ce->app, FALSE); - ce->in_async_call = TRUE; + gtk_widget_set_sensitive (ce->app, FALSE); + ce->in_async_call = TRUE; + if (ce->source_book != ce->target_book) { + /* Two-step move; add to target, then remove from source */ + eab_merging_book_add_contact (ce->target_book, ce->contact, + (EBookIdCallback) contact_added_cb, ecs); + } else { if (ce->is_new_contact) - eab_merging_book_add_contact (ce->book, ce->contact, (EBookIdCallback)contact_added_cb, ecs); + eab_merging_book_add_contact (ce->target_book, ce->contact, + (EBookIdCallback) contact_added_cb, ecs); else - eab_merging_book_commit_contact (ce->book, ce->contact, (EBookCallback)contact_modified_cb, ecs); + eab_merging_book_commit_contact (ce->target_book, ce->contact, + (EBookCallback) contact_modified_cb, ecs); } } @@ -1192,11 +1285,11 @@ delete_cb (GtkWidget *widget, gpointer data) extract_info (ce); - if (!ce->is_new_contact && ce->book) { + if (!ce->is_new_contact && ce->source_book) { gtk_widget_set_sensitive (ce->app, FALSE); ce->in_async_call = TRUE; - e_book_async_remove_contact (ce->book, contact, (EBookCallback)contact_deleted_cb, ce); + e_book_async_remove_contact (ce->source_book, contact, (EBookCallback)contact_deleted_cb, ce); } } @@ -1415,6 +1508,10 @@ e_contact_editor_init (EContactEditor *e_contact_editor) if (widget && GTK_IS_BUTTON(widget)) g_signal_connect (widget, "clicked", G_CALLBACK (categories_clicked), e_contact_editor); + widget = glade_xml_get_widget (e_contact_editor->gui, "source-option-menu-source"); + if (widget && E_IS_SOURCE_OPTION_MENU (widget)) + g_signal_connect (widget, "source_selected", + G_CALLBACK (source_selected), e_contact_editor); /* Construct the app */ bonobo_win = bonobo_window_new ("contact-editor-dialog", _("Contact Editor")); @@ -1520,9 +1617,14 @@ e_contact_editor_dispose (GObject *object) { e_contact_editor->contact = NULL; } - if (e_contact_editor->book) { - g_object_unref(e_contact_editor->book); - e_contact_editor->book = NULL; + if (e_contact_editor->source_book) { + g_object_unref(e_contact_editor->source_book); + e_contact_editor->source_book = NULL; + } + + if (e_contact_editor->target_book) { + g_object_unref(e_contact_editor->target_book); + e_contact_editor->target_book = NULL; } if (e_contact_editor->name) { @@ -1605,7 +1707,7 @@ e_contact_editor_new (EBook *book, gtk_object_sink (GTK_OBJECT (ce)); g_object_set (ce, - "book", book, + "source_book", book, "contact", contact, "is_new_contact", is_new_contact, "editable", editable, @@ -1625,11 +1727,33 @@ e_contact_editor_set_property (GObject *object, guint prop_id, const GValue *val editor = E_CONTACT_EDITOR (object); switch (prop_id){ - case PROP_BOOK: - if (editor->book) - g_object_unref(editor->book); - editor->book = E_BOOK(g_value_get_object (value)); - g_object_ref (editor->book); + case PROP_SOURCE_BOOK: + if (editor->source_book) + g_object_unref(editor->source_book); + editor->source_book = E_BOOK (g_value_get_object (value)); + g_object_ref (editor->source_book); + + if (!editor->target_book) { + editor->target_book = editor->source_book; + g_object_ref (editor->target_book); + } + + /* XXX more here about editable/etc. */ + break; + case PROP_TARGET_BOOK: + if (editor->target_book) + g_object_unref(editor->target_book); + editor->target_book = E_BOOK (g_value_get_object (value)); + g_object_ref (editor->target_book); + + e_book_async_get_supported_fields (editor->target_book, + (EBookFieldsCallback) supported_fields_cb, editor); + + if (!editor->changed && !editor->is_new_contact) { + editor->changed = TRUE; + command_state_changed (editor); + } + /* XXX more here about editable/etc. */ break; case PROP_CONTACT: @@ -1691,8 +1815,12 @@ e_contact_editor_get_property (GObject *object, guint prop_id, GValue *value, GP e_contact_editor = E_CONTACT_EDITOR (object); switch (prop_id) { - case PROP_BOOK: - g_value_set_object (value, e_contact_editor->book); + case PROP_SOURCE_BOOK: + g_value_set_object (value, e_contact_editor->source_book); + break; + + case PROP_TARGET_BOOK: + g_value_set_object (value, e_contact_editor->target_book); break; case PROP_CONTACT: @@ -2041,8 +2169,11 @@ set_source_field (EContactEditor *editor) GtkWidget *source_menu; ESource *source; + if (!editor->target_book) + return; + source_menu = glade_xml_get_widget (editor->gui, "source-option-menu-source"); - source = e_book_get_source (editor->book); + source = e_book_get_source (editor->target_book); e_source_option_menu_select (E_SOURCE_OPTION_MENU (source_menu), source); } diff --git a/addressbook/gui/contact-editor/e-contact-editor.h b/addressbook/gui/contact-editor/e-contact-editor.h index 7b404945f8..497d6d3a2a 100644 --- a/addressbook/gui/contact-editor/e-contact-editor.h +++ b/addressbook/gui/contact-editor/e-contact-editor.h @@ -54,7 +54,8 @@ struct _EContactEditor GtkObject object; /* item specific fields */ - EBook *book; + EBook *source_book; + EBook *target_book; EContact *contact; /* UI handler */ |