]> www.fi.muni.cz Git - evince.git/blobdiff - shell/ev-window.c
Move attachments bar to sidebar.
[evince.git] / shell / ev-window.c
index 905fd51b16ef823df449527432134b753cd9522f..1bfb26f636c05b491015de20d90d17051b32a8c6 100644 (file)
 #include "ev-page-action.h"
 #include "ev-sidebar.h"
 #include "ev-sidebar-links.h"
+#include "ev-sidebar-attachments.h"
 #include "ev-sidebar-thumbnails.h"
 #include "ev-view.h"
 #include "ev-password.h"
 #include "ev-password-view.h"
-#include "ev-attachment-bar.h"
 #include "ev-properties-dialog.h"
 #include "ev-ps-exporter.h"
 #include "ev-document-thumbnails.h"
@@ -90,7 +90,9 @@ typedef enum {
        EV_CHROME_TOOLBAR       = 1 << 1,
        EV_CHROME_FINDBAR       = 1 << 2,
        EV_CHROME_RAISE_TOOLBAR = 1 << 3,
-       EV_CHROME_NORMAL        = EV_CHROME_MENUBAR | EV_CHROME_TOOLBAR
+       EV_CHROME_FULLSCREEN_TOOLBAR    = 1 << 4,
+       EV_CHROME_SIDEBAR       = 1 << 5,
+       EV_CHROME_NORMAL        = EV_CHROME_MENUBAR | EV_CHROME_TOOLBAR | EV_CHROME_SIDEBAR
 } EvChrome;
 
 struct _EvWindowPrivate {
@@ -109,7 +111,7 @@ struct _EvWindowPrivate {
        GtkWidget *password_view;
        GtkWidget *sidebar_thumbs;
        GtkWidget *sidebar_links;
-       GtkWidget *attachment_bar;
+       GtkWidget *sidebar_attachments;
 
        /* Dialogs */
        GtkWidget *properties;
@@ -127,7 +129,7 @@ struct _EvWindowPrivate {
        /* Fullscreen mode */
        GtkWidget *fullscreen_toolbar;
        GtkWidget *fullscreen_popup;
-       GSource   *fullscreen_timeout_source;
+       guint      fullscreen_timeout_id;
        
        /* Popup link */
        GtkWidget *view_popup;
@@ -135,11 +137,12 @@ struct _EvWindowPrivate {
 
        /* Popup attachment */
        GtkWidget    *attachment_popup;
-       EvAttachment *attachment;
+       GList        *attach_list;
 
        /* Document */
        char *uri;
        char *local_uri;
+       EvLinkDest *dest;
        
        EvDocument *document;
        EvDocument *password_document;
@@ -174,6 +177,7 @@ static const GtkTargetEntry ev_drop_types[] = {
 #define SIDEBAR_DEFAULT_SIZE    132
 #define LINKS_SIDEBAR_ID "links"
 #define THUMBNAILS_SIDEBAR_ID "thumbnails"
+#define ATTACHMENTS_SIDEBAR_ID "attachments"
 
 static void    ev_window_update_actions                (EvWindow *ev_window);
 static void     ev_window_update_fullscreen_popup       (EvWindow         *window);
@@ -200,7 +204,6 @@ static void     ev_window_run_presentation              (EvWindow         *windo
 static void     ev_window_stop_presentation             (EvWindow         *window);
 static void     ev_window_cmd_view_presentation         (GtkAction        *action,
                                                         EvWindow         *window);
-static void     show_fullscreen_popup                   (EvWindow         *window);
 static void     ev_view_popup_cmd_open_link             (GtkAction        *action,
                                                         EvWindow         *window);
 static void     ev_view_popup_cmd_copy_link_address     (GtkAction        *action,
@@ -213,6 +216,13 @@ static void        ev_window_cmd_view_best_fit             (GtkAction        *action,
                                                         EvWindow         *ev_window);
 static void    ev_window_cmd_view_page_width           (GtkAction        *action, 
                                                         EvWindow         *ev_window);
+static void    fullscreen_set_timeout                  (EvWindow *window);
+static gboolean fullscreen_motion_notify_cb            (GtkWidget *widget,
+                                                        GdkEventMotion *event,
+                                                        gpointer user_data);
+static gboolean fullscreen_leave_notify_cb             (GtkWidget *widget,
+                                                        GdkEventCrossing *event,
+                                                        gpointer user_data);
 
 G_DEFINE_TYPE (EvWindow, ev_window, GTK_TYPE_WINDOW)
 
@@ -335,6 +345,8 @@ ev_window_update_actions (EvWindow *ev_window)
        ev_window_set_action_sensitive (ev_window, "EditCopy", has_pages && ev_view_get_has_selection (view));
        ev_window_set_action_sensitive (ev_window, "EditFindNext",
                              ev_view_can_find_next (view));
+       ev_window_set_action_sensitive (ev_window, "EditFindPrevious",
+                             ev_view_can_find_previous (view));
 
        ev_window_set_action_sensitive (ev_window, "ViewZoomIn",
                              has_pages && ev_view_can_zoom_in (view));
@@ -407,7 +419,7 @@ static void
 update_chrome_visibility (EvWindow *window)
 {
        EvWindowPrivate *priv = window->priv;
-       gboolean menubar, toolbar, findbar, fullscreen_toolbar;
+       gboolean menubar, toolbar, findbar, fullscreen_toolbar, sidebar;
        gboolean fullscreen_mode, presentation, fullscreen;
 
        presentation = ev_view_get_presentation (EV_VIEW (priv->view));
@@ -417,24 +429,22 @@ update_chrome_visibility (EvWindow *window)
        menubar = (priv->chrome & EV_CHROME_MENUBAR) != 0 && !fullscreen_mode;
        toolbar = ((priv->chrome & EV_CHROME_TOOLBAR) != 0  || 
                   (priv->chrome & EV_CHROME_RAISE_TOOLBAR) != 0) && !fullscreen_mode;
-       fullscreen_toolbar = ((priv->chrome & EV_CHROME_TOOLBAR) != 0 ||
-                             (priv->chrome & EV_CHROME_RAISE_TOOLBAR) != 0);
+       fullscreen_toolbar = ((priv->chrome & EV_CHROME_FULLSCREEN_TOOLBAR) != 0 || 
+                             (priv->chrome & EV_CHROME_RAISE_TOOLBAR) != 0) && fullscreen;
        findbar = (priv->chrome & EV_CHROME_FINDBAR) != 0;
+       sidebar = (priv->chrome & EV_CHROME_SIDEBAR) != 0 && !fullscreen_mode;
 
        set_widget_visibility (priv->menubar, menubar); 
        set_widget_visibility (priv->toolbar_dock, toolbar);
        set_widget_visibility (priv->find_bar, findbar);
+       set_widget_visibility (priv->sidebar, sidebar);
 
        ev_window_set_action_sensitive (window, "EditToolbar", toolbar);
        gtk_widget_set_sensitive (priv->menubar, menubar);
 
        if (priv->fullscreen_popup != NULL) {
-               if (fullscreen)
-                       show_fullscreen_popup (window);
-               else
-                       set_widget_visibility (priv->fullscreen_popup, FALSE);
-
                set_widget_visibility (priv->fullscreen_toolbar, fullscreen_toolbar);
+               set_widget_visibility (priv->fullscreen_popup, fullscreen_toolbar);
        }
 }
 
@@ -468,18 +478,17 @@ update_sizing_buttons (EvWindow *window)
                      NULL);
 
        switch (sizing_mode) {
-       case EV_SIZING_BEST_FIT:
-               best_fit = TRUE;
-               page_width = FALSE;
-               break;
-       case EV_SIZING_FIT_WIDTH:
-               best_fit = FALSE;
-               page_width = TRUE;
-               break;
-
-       default:
-               best_fit = page_width = FALSE;
-               break;
+               case EV_SIZING_BEST_FIT:
+                       best_fit = TRUE;
+                       page_width = FALSE;
+                       break;
+               case EV_SIZING_FIT_WIDTH:
+                       best_fit = FALSE;
+                       page_width = TRUE;
+                       break;
+               default:
+                       best_fit = page_width = FALSE;
+                       break;
        }
 
        action = gtk_action_group_get_action (action_group, "ViewBestFit");
@@ -580,26 +589,6 @@ update_document_mode (EvWindow *window, EvDocumentMode mode)
        }
 }
 
-static void
-update_sidebar_visibility (EvWindow *window)
-{
-       char *uri = window->priv->uri;
-       GValue sidebar_visibility = { 0, };
-
-       if (ev_metadata_manager_get (uri, "sidebar_visibility", &sidebar_visibility, FALSE)) {
-               set_widget_visibility (window->priv->sidebar,
-                                      g_value_get_boolean (&sidebar_visibility));
-       }
-}
-
-static void
-update_attachment_bar_visibility (EvWindow *window)
-{
-       set_widget_visibility (window->priv->attachment_bar,
-                              (window->priv->document &&
-                              ev_document_has_attachments (window->priv->document)));
-}
-
 static void
 setup_document_from_metadata (EvWindow *window)
 {
@@ -635,8 +624,10 @@ setup_sidebar_from_metadata (EvWindow *window, EvDocument *document)
        GtkWidget *sidebar = window->priv->sidebar;
        GtkWidget *links = window->priv->sidebar_links;
        GtkWidget *thumbs = window->priv->sidebar_thumbs;
+       GtkWidget *attachments = window->priv->sidebar_attachments;
        GValue sidebar_size = { 0, };
        GValue sidebar_page = { 0, };
+       GValue sidebar_visibility = { 0, };
 
        if (ev_metadata_manager_get (uri, "sidebar_size", &sidebar_size, FALSE)) {
                gtk_paned_set_position (GTK_PANED (window->priv->hpaned),
@@ -646,9 +637,11 @@ setup_sidebar_from_metadata (EvWindow *window, EvDocument *document)
        if (ev_metadata_manager_get (uri, "sidebar_page", &sidebar_page, FALSE)) {
                const char *page_id = g_value_get_string (&sidebar_page);
 
-               if (strcmp (page_id, "links") == 0 && ev_sidebar_page_support_document (EV_SIDEBAR_PAGE (links), document)) {
+               if (strcmp (page_id, LINKS_SIDEBAR_ID) == 0 && ev_sidebar_page_support_document (EV_SIDEBAR_PAGE (links), document)) {
                        ev_sidebar_set_page (EV_SIDEBAR (sidebar), links);
-               } else if (strcmp (page_id, "thumbnails") && ev_sidebar_page_support_document (EV_SIDEBAR_PAGE (thumbs), document)) {
+               } else if (strcmp (page_id, THUMBNAILS_SIDEBAR_ID) && ev_sidebar_page_support_document (EV_SIDEBAR_PAGE (thumbs), document)) {
+                       ev_sidebar_set_page (EV_SIDEBAR (sidebar), thumbs);
+               } else if (strcmp (page_id, ATTACHMENTS_SIDEBAR_ID) && ev_sidebar_page_support_document (EV_SIDEBAR_PAGE (attachments), document)) {
                        ev_sidebar_set_page (EV_SIDEBAR (sidebar), thumbs);
                }
        } else {
@@ -657,7 +650,9 @@ setup_sidebar_from_metadata (EvWindow *window, EvDocument *document)
                }
        }
 
-       update_sidebar_visibility (window);
+       if (ev_metadata_manager_get (uri, "sidebar_visibility", &sidebar_visibility, FALSE)) {
+               update_chrome_flag (window, EV_CHROME_SIDEBAR, g_value_get_boolean (&sidebar_visibility));
+       }
 }
 
 static void
@@ -747,17 +742,17 @@ setup_view_from_metadata (EvWindow *window)
        if (ev_metadata_manager_get (uri, "rotation", &rotation, TRUE)) {
                if (g_value_get_int (&rotation)) {
                        switch (g_value_get_int (&rotation)) {
-                       case 90:
-                               ev_view_set_rotation (view, 90);
-                               break;
-                       case 180:
-                               ev_view_set_rotation (view, 180);
-                               break;
-                       case 270:
-                               ev_view_set_rotation (view, 270);
-                               break;
-                       default:
-                               break;
+                               case 90:
+                                       ev_view_set_rotation (view, 90);
+                                       break;
+                               case 180:
+                                       ev_view_set_rotation (view, 180);
+                                       break;
+                               case 270:
+                                       ev_view_set_rotation (view, 270);
+                                       break;
+                               default:
+                                       break;
                        }
                }
        }
@@ -770,7 +765,6 @@ ev_window_setup_document (EvWindow *ev_window)
        EvDocument *document;
        EvView *view = EV_VIEW (ev_window->priv->view);
        EvSidebar *sidebar = EV_SIDEBAR (ev_window->priv->sidebar);
-       EvAttachmentBar *attachbar = EV_ATTACHMENT_BAR (ev_window->priv->attachment_bar);
        GtkAction *action;
 
        document = ev_window->priv->document;
@@ -791,9 +785,6 @@ ev_window_setup_document (EvWindow *ev_window)
        }
        ev_window_set_page_mode (ev_window, PAGE_MODE_DOCUMENT);
 
-       ev_attachment_bar_set_document (attachbar, document);
-       update_attachment_bar_visibility (ev_window);
-
        ev_window_title_set_document (ev_window->priv->title, document);
        ev_window_title_set_uri (ev_window->priv->title, ev_window->priv->uri);
 
@@ -953,6 +944,9 @@ ev_window_xfer_job_cb  (EvJobXfer *job,
                ev_window_setup_document (ev_window);
                ev_window_add_recent (ev_window, ev_window->priv->uri);         
                ev_window_clear_xfer_job (ev_window);
+
+               if (ev_window->priv->dest)
+                       ev_window_goto_dest (ev_window, ev_window->priv->dest);
                
                return;
        }
@@ -1014,10 +1008,14 @@ ev_window_close_dialogs (EvWindow *ev_window)
 }
 
 void
-ev_window_open_uri (EvWindow *ev_window, const char *uri)
+ev_window_open_uri (EvWindow *ev_window, const char *uri, EvLinkDest *dest)
 {
        g_free (ev_window->priv->uri);
        ev_window->priv->uri = NULL;
+
+       if (ev_window->priv->dest)
+               g_object_unref (ev_window->priv->dest);
+       ev_window->priv->dest = dest ? g_object_ref (dest) : NULL;
        
        ev_window_close_dialogs (ev_window);
        ev_window_clear_xfer_job (ev_window);
@@ -1032,6 +1030,12 @@ ev_window_open_uri (EvWindow *ev_window, const char *uri)
        ev_job_queue_add_job (ev_window->priv->xfer_job, EV_JOB_PRIORITY_HIGH);
 }
 
+void
+ev_window_goto_dest (EvWindow *ev_window, EvLinkDest *dest)
+{
+       ev_view_goto_dest (EV_VIEW (ev_window->priv->view), dest);
+}
+
 static void
 file_open_dialog_response_cb (GtkWidget *chooser,
                              gint       response_id,
@@ -1135,7 +1139,7 @@ file_save_dialog_response_cb (GtkWidget *fc,
        gboolean success;
 
        if (response_id == GTK_RESPONSE_OK) {
-               const char *uri;
+               gchar *uri;
                GError *err = NULL;
 
                uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (fc));
@@ -1150,6 +1154,8 @@ file_save_dialog_response_cb (GtkWidget *fc,
                        ev_window_error_dialog (GTK_WINDOW (fc), msg, err);
                        g_free (msg);
                }
+
+               g_free (uri);
        }
 
        gtk_widget_destroy (fc);
@@ -1446,6 +1452,14 @@ ev_window_cmd_edit_find_next (GtkAction *action, EvWindow *ev_window)
        ev_view_find_next (EV_VIEW (ev_window->priv->view));
 }
 
+static void
+ev_window_cmd_edit_find_previous (GtkAction *action, EvWindow *ev_window)
+{
+        g_return_if_fail (EV_IS_WINDOW (ev_window));
+
+       ev_view_find_previous (EV_VIEW (ev_window->priv->view));
+}
+
 static void
 ev_window_cmd_edit_copy (GtkAction *action, EvWindow *ev_window)
 {
@@ -1468,7 +1482,7 @@ ev_window_update_fullscreen_popup (EvWindow *window)
        if (GTK_WIDGET (window)->window == NULL)
                return;
 
-       toolbar = (window->priv->chrome & EV_CHROME_TOOLBAR) != 0 || 
+       toolbar = (window->priv->chrome & EV_CHROME_FULLSCREEN_TOOLBAR) != 0 || 
                  (window->priv->chrome & EV_CHROME_RAISE_TOOLBAR) != 0;
        popup_width = popup->requisition.width;
        popup_height = popup->requisition.height;
@@ -1548,11 +1562,10 @@ static gboolean
 fullscreen_timeout_cb (gpointer data)
 {
        EvWindow *window = EV_WINDOW (data);
-
-       g_object_set (window->priv->fullscreen_popup, "visible", FALSE, NULL);
+       
+       update_chrome_flag (window, EV_CHROME_FULLSCREEN_TOOLBAR, FALSE);
        ev_view_hide_cursor (EV_VIEW (window->priv->view));
-       g_source_unref (window->priv->fullscreen_timeout_source);
-       window->priv->fullscreen_timeout_source = NULL;
+       window->priv->fullscreen_timeout_id = 0;
 
        return FALSE;
 }
@@ -1560,43 +1573,31 @@ fullscreen_timeout_cb (gpointer data)
 static void
 fullscreen_set_timeout (EvWindow *window)
 {
-       GSource *source;
-
-       if (window->priv->fullscreen_timeout_source != NULL) {
-               g_source_unref (window->priv->fullscreen_timeout_source);
-               g_source_destroy (window->priv->fullscreen_timeout_source);
+       if (window->priv->fullscreen_timeout_id != 0) {
+               g_source_remove (window->priv->fullscreen_timeout_id);
        }
+       
+       window->priv->fullscreen_timeout_id = 
+           g_timeout_add (FULLSCREEN_TIMEOUT, fullscreen_timeout_cb, window);
 
-       source = g_timeout_source_new (FULLSCREEN_TIMEOUT);
-       g_source_set_callback (source, fullscreen_timeout_cb, window, NULL);
-       g_source_attach (source, NULL);
-       window->priv->fullscreen_timeout_source = source;
+       update_chrome_flag (window, EV_CHROME_FULLSCREEN_TOOLBAR, TRUE);
+       update_chrome_visibility (window);
+       ev_view_show_cursor (EV_VIEW (window->priv->view));
 }
 
 static void
 fullscreen_clear_timeout (EvWindow *window)
 {
-       if (window->priv->fullscreen_timeout_source != NULL) {
-               g_source_unref (window->priv->fullscreen_timeout_source);
-               g_source_destroy (window->priv->fullscreen_timeout_source);
+       if (window->priv->fullscreen_timeout_id != 0) {
+               g_source_remove (window->priv->fullscreen_timeout_id);
        }
-
-       window->priv->fullscreen_timeout_source = NULL;
+       
+       window->priv->fullscreen_timeout_id = 0;
+       update_chrome_visibility (window);
        ev_view_show_cursor (EV_VIEW (window->priv->view));
 }
 
 
-static void
-show_fullscreen_popup (EvWindow *window)
-{
-       if (!GTK_WIDGET_VISIBLE (window->priv->fullscreen_popup)) {
-               g_object_set (window->priv->fullscreen_popup, "visible", TRUE, NULL);
-               ev_view_show_cursor (EV_VIEW (window->priv->view));
-       }
-
-       fullscreen_set_timeout (window);        
-}
-
 static gboolean
 fullscreen_motion_notify_cb (GtkWidget *widget,
                             GdkEventMotion *event,
@@ -1604,7 +1605,7 @@ fullscreen_motion_notify_cb (GtkWidget *widget,
 {
        EvWindow *window = EV_WINDOW (user_data);
 
-       show_fullscreen_popup (window);
+       fullscreen_set_timeout (window);
 
        return FALSE;
 }
@@ -1704,19 +1705,21 @@ ev_window_update_fullscreen_action (EvWindow *window)
 static void
 ev_window_run_fullscreen (EvWindow *window)
 {
-       ev_view_set_fullscreen (EV_VIEW (window->priv->view), TRUE);
        if (window->priv->fullscreen_popup == NULL)
                window->priv->fullscreen_popup
                        = ev_window_create_fullscreen_popup (window);
 
-       update_chrome_visibility (window);
-       gtk_widget_hide (window->priv->sidebar);
-       gtk_widget_hide (window->priv->attachment_bar);
-
        g_object_set (G_OBJECT (window->priv->scrolled_window),
                      "shadow-type", GTK_SHADOW_NONE,
                      NULL);
+
+       ev_view_set_fullscreen (EV_VIEW (window->priv->view), TRUE);
+       ev_window_update_fullscreen_action (window);
        
+       gtk_window_fullscreen (GTK_WINDOW (window));
+       gtk_widget_grab_focus (window->priv->view);
+       ev_window_update_fullscreen_popup (window);
+
        g_signal_connect (window->priv->view,
                          "motion-notify-event",
                          G_CALLBACK (fullscreen_motion_notify_cb),
@@ -1727,11 +1730,6 @@ ev_window_run_fullscreen (EvWindow *window)
                          window);
        fullscreen_set_timeout (window);
 
-       gtk_widget_grab_focus (window->priv->view);
-       ev_window_update_fullscreen_action (window);
-       gtk_window_fullscreen (GTK_WINDOW (window));
-       ev_window_update_fullscreen_popup (window);
-
        if (!ev_window_is_empty (window))
                ev_metadata_manager_set_boolean (window->priv->uri, "fullscreen", TRUE);
 }
@@ -1744,24 +1742,21 @@ ev_window_stop_fullscreen (EvWindow *window)
        if (!ev_view_get_fullscreen (EV_VIEW (view)))
                return;
 
-       ev_view_set_fullscreen (view, FALSE);
        g_object_set (G_OBJECT (window->priv->scrolled_window),
                      "shadow-type", GTK_SHADOW_IN,
                      NULL);
 
-       fullscreen_clear_timeout (window);
+       ev_view_set_fullscreen (view, FALSE);
+       ev_window_update_fullscreen_action (window);
+       gtk_window_unfullscreen (GTK_WINDOW (window));
 
-       g_signal_handlers_disconnect_by_func (view,
+       g_signal_handlers_disconnect_by_func (window->priv->view,
                                              (gpointer) fullscreen_motion_notify_cb,
                                              window);
-       g_signal_handlers_disconnect_by_func (view,
+       g_signal_handlers_disconnect_by_func (window->priv->view,
                                              (gpointer) fullscreen_leave_notify_cb,
                                              window);
-       ev_window_update_fullscreen_action (window);
-       gtk_window_unfullscreen (GTK_WINDOW (window));
-       update_chrome_visibility (window);
-       update_sidebar_visibility (window);
-       update_attachment_bar_visibility (window);
+       fullscreen_clear_timeout (window);
 
        if (!ev_window_is_empty (window))
                ev_metadata_manager_set_boolean (window->priv->uri, "fullscreen", FALSE);
@@ -1804,13 +1799,21 @@ ev_window_run_presentation (EvWindow *window)
                      "shadow-type", GTK_SHADOW_NONE,
                      NULL);
 
-       gtk_widget_grab_focus (window->priv->view);
        ev_view_set_presentation (EV_VIEW (window->priv->view), TRUE);
-       gtk_window_fullscreen (GTK_WINDOW (window));
        ev_window_update_presentation_action (window);
-       update_chrome_visibility (window);
-       gtk_widget_hide (window->priv->sidebar);
-       gtk_widget_hide (window->priv->attachment_bar);
+
+       gtk_widget_grab_focus (window->priv->view);
+       gtk_window_fullscreen (GTK_WINDOW (window));
+
+       g_signal_connect (window->priv->view,
+                         "motion-notify-event",
+                         G_CALLBACK (fullscreen_motion_notify_cb),
+                         window);
+       g_signal_connect (window->priv->view,
+                         "leave-notify-event",
+                         G_CALLBACK (fullscreen_leave_notify_cb),
+                         window);
+       fullscreen_set_timeout (window);
 
        if (!ev_window_is_empty (window))
                ev_metadata_manager_set_boolean (window->priv->uri, "presentation", TRUE);
@@ -1825,12 +1828,18 @@ ev_window_stop_presentation (EvWindow *window)
        g_object_set (G_OBJECT (window->priv->scrolled_window),
                      "shadow-type", GTK_SHADOW_IN,
                      NULL);
+
        ev_view_set_presentation (EV_VIEW (window->priv->view), FALSE);
-       gtk_window_unfullscreen (GTK_WINDOW (window));
        ev_window_update_presentation_action (window);
-       update_chrome_visibility (window);
-       update_sidebar_visibility (window);
-       update_attachment_bar_visibility (window);
+       gtk_window_unfullscreen (GTK_WINDOW (window));
+
+       g_signal_handlers_disconnect_by_func (window->priv->view,
+                                             (gpointer) fullscreen_motion_notify_cb,
+                                             window);
+       g_signal_handlers_disconnect_by_func (window->priv->view,
+                                             (gpointer) fullscreen_leave_notify_cb,
+                                             window);
+       fullscreen_clear_timeout (window);
 
        if (!ev_window_is_empty (window))
                ev_metadata_manager_set_boolean (window->priv->uri, "presentation", FALSE);
@@ -1857,14 +1866,9 @@ ev_window_focus_in_event (GtkWidget *widget, GdkEventFocus *event)
 {
        EvWindow *window = EV_WINDOW (widget);
        EvWindowPrivate *priv = window->priv;
-       gboolean fullscreen;
 
-       g_object_get (priv->view,
-                     "fullscreen", &fullscreen,
-                     NULL);
-
-       if (fullscreen)
-               show_fullscreen_popup (window);
+       if (ev_view_get_fullscreen (EV_VIEW (priv->view)))
+               fullscreen_set_timeout (window);
 
        return GTK_WIDGET_CLASS (ev_window_parent_class)->focus_in_event (widget, event);
 }
@@ -1874,14 +1878,11 @@ ev_window_focus_out_event (GtkWidget *widget, GdkEventFocus *event)
 {
        EvWindow *window = EV_WINDOW (widget);
        EvWindowPrivate *priv = window->priv;
-       gboolean fullscreen;
-
-       g_object_get (priv->view,
-                     "fullscreen", &fullscreen,
-                     NULL);
 
-       if (fullscreen)
+       if (ev_view_get_fullscreen (EV_VIEW (priv->view))) {
+               fullscreen_set_timeout (window);
                gtk_widget_hide (priv->fullscreen_popup);
+       }
 
        return GTK_WIDGET_CLASS (ev_window_parent_class)->focus_out_event (widget, event);
 }
@@ -1925,14 +1926,14 @@ ev_window_set_page_mode (EvWindow         *window,
        window->priv->page_mode = page_mode;
 
        switch (page_mode) {
-       case PAGE_MODE_DOCUMENT:
-               child = window->priv->view;
-               break;
-       case PAGE_MODE_PASSWORD:
-               child = window->priv->password_view;
-               break;
-       default:
-               g_assert_not_reached ();
+               case PAGE_MODE_DOCUMENT:
+                       child = window->priv->view;
+                       break;
+               case PAGE_MODE_PASSWORD:
+                       child = window->priv->password_view;
+                       break;
+               default:
+                       g_assert_not_reached ();
        }
 
        real_child = gtk_bin_get_child (GTK_BIN (window->priv->scrolled_window));
@@ -2095,7 +2096,7 @@ ev_window_cmd_view_reload (GtkAction *action, EvWindow *ev_window)
        page = ev_page_cache_get_current_page (ev_window->priv->page_cache);
        uri = g_strdup (ev_window->priv->uri);
 
-       ev_window_open_uri (ev_window, uri);
+       ev_window_open_uri (ev_window, uri, NULL);
 
        /* In case the number of pages in the document has changed. */
        page = CLAMP (page, 0, ev_page_cache_get_n_pages (ev_window->priv->page_cache) - 1);
@@ -2219,30 +2220,30 @@ ev_window_sizing_mode_changed_cb (EvView *view, GParamSpec *pspec,
                update_view_size (NULL, ev_window);
 
        switch (sizing_mode) {
-       case EV_SIZING_BEST_FIT:
-               g_object_set (G_OBJECT (scrolled_window),
-                             "hscrollbar-policy", GTK_POLICY_NEVER,
-                             "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
-                             NULL);
-               g_signal_connect (ev_window->priv->view, "zoom_invalid",
-                                 G_CALLBACK (update_view_size),
-                                 ev_window);
-               break;
-       case EV_SIZING_FIT_WIDTH:
-               g_object_set (G_OBJECT (scrolled_window),
-                             "hscrollbar-policy", GTK_POLICY_NEVER,
-                             "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
-                             NULL);
-               g_signal_connect (ev_window->priv->view, "zoom_invalid",
-                                 G_CALLBACK (update_view_size),
-                                 ev_window);
-               break;
-       case EV_SIZING_FREE:
-               g_object_set (G_OBJECT (scrolled_window),
-                             "hscrollbar-policy", GTK_POLICY_AUTOMATIC,
-                             "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
-                             NULL);
-               break;
+               case EV_SIZING_BEST_FIT:
+                       g_object_set (G_OBJECT (scrolled_window),
+                                     "hscrollbar-policy", GTK_POLICY_NEVER,
+                                     "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
+                                     NULL);
+                       g_signal_connect (ev_window->priv->view, "zoom_invalid",
+                                         G_CALLBACK (update_view_size),
+                                         ev_window);
+                       break;
+               case EV_SIZING_FIT_WIDTH:
+                       g_object_set (G_OBJECT (scrolled_window),
+                                     "hscrollbar-policy", GTK_POLICY_NEVER,
+                                     "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
+                                     NULL);
+                       g_signal_connect (ev_window->priv->view, "zoom_invalid",
+                                         G_CALLBACK (update_view_size),
+                                         ev_window);
+                       break;
+               case EV_SIZING_FREE:
+                       g_object_set (G_OBJECT (scrolled_window),
+                                     "hscrollbar-policy", GTK_POLICY_AUTOMATIC,
+                                     "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
+                                     NULL);
+                       break;
        }
 
        update_sizing_buttons (ev_window);
@@ -2416,6 +2417,7 @@ ev_window_cmd_help_about (GtkAction *action, EvWindow *ev_window)
                "authors", authors,
                "documenters", documenters,
                "translator-credits", _("translator-credits"),
+               "logo-icon-name", "evince",
                NULL);
 
        g_free (comments);
@@ -2435,8 +2437,8 @@ ev_window_view_toolbar_cb (GtkAction *action, EvWindow *ev_window)
 static void
 ev_window_view_sidebar_cb (GtkAction *action, EvWindow *ev_window)
 {
-       set_widget_visibility (ev_window->priv->sidebar,
-                              gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)));
+       update_chrome_flag (ev_window, EV_CHROME_SIDEBAR,
+                           gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)));
 }
 
 static void
@@ -2453,6 +2455,8 @@ ev_window_sidebar_current_page_changed_cb (EvSidebar  *ev_sidebar,
                id = LINKS_SIDEBAR_ID;
        } else if (current_page == ev_window->priv->sidebar_thumbs) {
                id = THUMBNAILS_SIDEBAR_ID;
+       } else if (current_page == ev_window->priv->sidebar_attachments) {
+               id = ATTACHMENTS_SIDEBAR_ID;
        } else {
                g_assert_not_reached();
        }
@@ -2464,20 +2468,16 @@ ev_window_sidebar_current_page_changed_cb (EvSidebar  *ev_sidebar,
 }
 
 static void
-ev_window_sidebar_visibility_changed_cb (EvSidebar *ev_sidebar, GParamSpec *pspec,
+ev_window_sidebar_visibility_changed_cb (EvSidebar  *ev_sidebar,
+                                        GParamSpec *pspec,
                                         EvWindow   *ev_window)
 {
        EvView *view = EV_VIEW (ev_window->priv->view);
        GtkAction *action;
 
        action = gtk_action_group_get_action (ev_window->priv->action_group, "ViewSidebar");
-       
-       g_signal_handlers_block_by_func
-               (action, G_CALLBACK (ev_window_view_sidebar_cb), ev_window);
        gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action),
                                      GTK_WIDGET_VISIBLE (ev_sidebar));
-       g_signal_handlers_unblock_by_func
-               (action, G_CALLBACK (ev_window_view_sidebar_cb), ev_window);
 
        if (!ev_view_get_presentation (view) && 
            !ev_view_get_fullscreen (view) &&
@@ -2487,22 +2487,10 @@ ev_window_sidebar_visibility_changed_cb (EvSidebar *ev_sidebar, GParamSpec *pspe
        }
 }
 
-static void
-ev_window_attachment_bar_toggled_cb (EvAttachmentBar *ev_attachbar,
-                                    GParamSpec      *param_spec,
-                                    EvWindow        *ev_window)
-{
-       if (gtk_expander_get_expanded (GTK_EXPANDER (ev_attachbar))) {
-               gtk_widget_grab_focus (GTK_WIDGET (ev_attachbar));
-       } else {
-               gtk_widget_grab_focus (ev_window->priv->view);
-       }
-}
-
 static gboolean
-view_menu_popup_cb (EvView         *view,
-                   EvLink         *link,
-                   EvWindow       *ev_window)
+view_menu_popup_cb (EvView   *view,
+                   EvLink   *link,
+                   EvWindow *ev_window)
 {
        GtkWidget *popup;
        gboolean   show_external = FALSE;
@@ -2514,6 +2502,7 @@ view_menu_popup_cb (EvView         *view,
        
        if (ev_window->priv->link)
                g_object_unref (ev_window->priv->link);
+       
        if (link)
                ev_window->priv->link = g_object_ref (link);
        else    
@@ -2521,23 +2510,26 @@ view_menu_popup_cb (EvView         *view,
 
        popup = ev_window->priv->view_popup;
 
-       if (ev_window->priv->link) 
-               switch (ev_link_get_link_type (ev_window->priv->link)) {
-                       case EV_LINK_TYPE_PAGE:
-                       case EV_LINK_TYPE_PAGE_FIT:
-                       case EV_LINK_TYPE_PAGE_FITH:
-                       case EV_LINK_TYPE_PAGE_FITV:
-                       case EV_LINK_TYPE_PAGE_FITR:
-                       case EV_LINK_TYPE_PAGE_XYZ:
+       if (ev_window->priv->link) {
+               EvLinkAction *ev_action;
+
+               ev_action = ev_link_get_action (link);
+               if (!ev_action)
+                       return FALSE;
+               
+               switch (ev_link_action_get_action_type (ev_action)) {
+                       case EV_LINK_ACTION_TYPE_GOTO_DEST:
+                       case EV_LINK_ACTION_TYPE_GOTO_REMOTE:
                                show_internal = TRUE;
                                break;
-                       case EV_LINK_TYPE_EXTERNAL_URI:
-                       case EV_LINK_TYPE_LAUNCH:       
+                       case EV_LINK_ACTION_TYPE_EXTERNAL_URI:
+                       case EV_LINK_ACTION_TYPE_LAUNCH:
                                show_external = TRUE;
                                break;
-                       default:
+                       default:
                                break;
                }
+       }
        
        action = gtk_action_group_get_action (ev_window->priv->view_popup_action_group,
                                              "OpenLink");
@@ -2554,28 +2546,33 @@ view_menu_popup_cb (EvView         *view,
        gtk_menu_popup (GTK_MENU (popup), NULL, NULL,
                        NULL, NULL,
                        3, gtk_get_current_event_time ());
-       return FALSE;
+       return TRUE;
 }
 
-static void
-attachment_bar_menu_popup_cb (EvAttachmentBar *attachbar,
-                             EvAttachment    *attachment,
+static gboolean
+attachment_bar_menu_popup_cb (EvSidebarAttachments *attachbar,
+                             GList           *attach_list,
                              EvWindow        *ev_window)
 {
        GtkWidget *popup;
 
-       if (ev_window->priv->attachment)
-               g_object_unref (ev_window->priv->attachment);
-       if (attachment)
-               ev_window->priv->attachment = g_object_ref (attachment);
-       else
-               ev_window->priv->attachment = NULL;
+       g_assert (attach_list != NULL);
+
+       if (ev_window->priv->attach_list) {
+               g_list_foreach (ev_window->priv->attach_list,
+                               (GFunc) g_object_unref, NULL);
+               g_list_free (ev_window->priv->attach_list);
+       }
+       
+       ev_window->priv->attach_list = attach_list;
        
        popup = ev_window->priv->attachment_popup;
 
        gtk_menu_popup (GTK_MENU (popup), NULL, NULL,
                        NULL, NULL,
                        3, gtk_get_current_event_time ());
+
+       return TRUE;
 }
 
 static void
@@ -2771,9 +2768,12 @@ ev_window_dispose (GObject *object)
                priv->link = NULL;
        }
 
-       if (priv->attachment) {
-               g_object_unref (priv->attachment);
-               priv->attachment = NULL;
+       if (priv->attach_list) {
+               g_list_foreach (priv->attach_list,
+                               (GFunc) g_object_unref,
+                               NULL);
+               g_list_free (priv->attach_list);
+               priv->attach_list = NULL;
        }
 
        if (priv->find_bar) {
@@ -2789,10 +2789,14 @@ ev_window_dispose (GObject *object)
                priv->uri = NULL;
        }
 
-       if (window->priv->fullscreen_timeout_source) {
-               g_source_unref (window->priv->fullscreen_timeout_source);
-               g_source_destroy (window->priv->fullscreen_timeout_source);
-               window->priv->fullscreen_timeout_source = NULL;
+       if (priv->dest) {
+               g_object_unref (priv->dest);
+               priv->dest = NULL;
+       }
+
+       if (priv->fullscreen_timeout_id) {
+               g_source_remove (priv->fullscreen_timeout_id);
+               priv->fullscreen_timeout_id = 0;
        }
        destroy_fullscreen_popup (window);
 
@@ -2843,11 +2847,13 @@ static const GtkActionEntry entries[] = {
           G_CALLBACK (ev_window_cmd_edit_copy) },
        { "EditSelectAll", NULL, N_("Select _All"), "<control>A", NULL,
          G_CALLBACK (ev_window_cmd_edit_select_all) },
-        { "EditFind", GTK_STOCK_FIND, NULL, "<control>F",
+        { "EditFind", GTK_STOCK_FIND, N_("_Find..."), "<control>F",
           N_("Find a word or phrase in the document"),
           G_CALLBACK (ev_window_cmd_edit_find) },
        { "EditFindNext", NULL, N_("Find Ne_xt"), "<control>G", NULL,
          G_CALLBACK (ev_window_cmd_edit_find_next) },
+       { "EditFindPrevious", NULL, N_("Find Pre_vious"), "<shift><control>G", NULL,
+         G_CALLBACK (ev_window_cmd_edit_find_previous) },
         { "EditToolbar", NULL, N_("T_oolbar"), NULL, NULL,
           G_CALLBACK (ev_window_cmd_edit_toolbar) },
        { "EditRotateLeft", NULL, N_("Rotate _Left"), NULL, NULL,
@@ -3015,9 +3021,7 @@ drag_data_received_cb (GtkWidget *widget, GdkDragContext *context,
 static void
 activate_link_cb (EvPageAction *page_action, EvLink *link, EvWindow *window)
 {
-       g_return_if_fail (EV_IS_WINDOW (window));
-
-       ev_view_goto_link (EV_VIEW (window->priv->view), link);
+       ev_view_handle_link (EV_VIEW (window->priv->view), link);
        gtk_widget_grab_focus (window->priv->view);
 }
 
@@ -3214,13 +3218,13 @@ window_configure_event_cb (EvWindow *window, GdkEventConfigure *event, gpointer
 static void
 sidebar_links_link_activated_cb (EvSidebarLinks *sidebar_links, EvLink *link, EvWindow *window)
 {
-       ev_view_goto_link (EV_VIEW (window->priv->view), link);
+       ev_view_handle_link (EV_VIEW (window->priv->view), link);
 }
 
 static void
-launch_link (EvWindow *window, EvLink *link)
+launch_action (EvWindow *window, EvLinkAction *action)
 {
-       const char *filename = ev_link_get_filename (link);
+       const char *filename = ev_link_action_get_filename (action);
        char *uri = NULL;
 
        if (filename  && g_path_is_absolute (filename)) {
@@ -3252,12 +3256,12 @@ launch_link (EvWindow *window, EvLink *link)
 }
 
 static void
-launch_external_uri (EvWindow *window, EvLink *link)
+launch_external_uri (EvWindow *window, EvLinkAction *action)
 {
        const char *uri;
        char *escaped;
 
-       uri = ev_link_get_uri (link);
+       uri = ev_link_action_get_uri (action);
        escaped = gnome_vfs_escape_host_and_path_string (uri);
 
        gnome_vfs_url_show (escaped);
@@ -3265,33 +3269,86 @@ launch_external_uri (EvWindow *window, EvLink *link)
 }
 
 static void
-view_external_link_cb (EvView *view, EvLink *link, EvWindow *window)
+open_remote_link (EvWindow *window, EvLinkAction *action)
+{
+       gchar *uri;
+       gchar *dir;
+
+       dir = g_path_get_dirname (window->priv->uri);
+       
+       uri = g_build_filename (dir, ev_link_action_get_filename (action),
+                               NULL);
+       g_free (dir);
+       
+       ev_application_open_uri_at_dest (EV_APP, uri,
+                                        ev_link_action_get_dest (action),
+                                        GDK_CURRENT_TIME);
+
+       g_free (uri);
+}
+
+static void
+do_action_named (EvWindow *window, EvLinkAction *action)
+{
+       const gchar *name = ev_link_action_get_name (action);
+
+       if (g_ascii_strcasecmp (name, "FirstPage") == 0) {
+               ev_window_cmd_go_first_page (NULL, window);
+       } else if (g_ascii_strcasecmp (name, "PrevPage") == 0) {
+               ev_window_cmd_go_previous_page (NULL, window);
+       } else if (g_ascii_strcasecmp (name, "NextPage") == 0) {
+               ev_window_cmd_go_next_page (NULL, window);
+       } else if (g_ascii_strcasecmp (name, "LastPage") == 0) {
+               ev_window_cmd_go_last_page (NULL, window);
+       } else if (g_ascii_strcasecmp (name, "GoToPage") == 0) {
+               ev_window_cmd_focus_page_selector (NULL, window);
+       } else if (g_ascii_strcasecmp (name, "Find") == 0) {
+               ev_window_cmd_edit_find (NULL, window);
+       } else {
+               g_warning ("Unimplemented named action: %s, please post a bug report with a testcase.",
+                          name);
+       }
+}
+
+static void
+view_external_link_cb (EvView *view, EvLinkAction *action, EvWindow *window)
 {
-       switch (ev_link_get_link_type (link)) {
-       case EV_LINK_TYPE_EXTERNAL_URI:
-               launch_external_uri (window, link);
-               break;
-       case EV_LINK_TYPE_LAUNCH:
-               launch_link (window, link);
-               break;
-       default:
-               g_assert_not_reached ();
+       switch (ev_link_action_get_action_type (action)) {
+               case EV_LINK_ACTION_TYPE_EXTERNAL_URI:
+                       launch_external_uri (window, action);
+                       break;
+               case EV_LINK_ACTION_TYPE_LAUNCH:
+                       launch_action (window, action);
+                       break;
+               case EV_LINK_ACTION_TYPE_GOTO_REMOTE:
+                       open_remote_link (window, action);
+                       break;
+               case EV_LINK_ACTION_TYPE_NAMED:
+                       do_action_named (window, action);
+                       break;
+               default:
+                       g_assert_not_reached ();
        }
 }
 
 static void
 ev_view_popup_cmd_open_link (GtkAction *action, EvWindow *window)
 {
-       ev_view_goto_link (EV_VIEW (window->priv->view), window->priv->link);
+       ev_view_handle_link (EV_VIEW (window->priv->view), window->priv->link);
 }
 
 static void
 ev_view_popup_cmd_copy_link_address (GtkAction *action, EvWindow *window)
 {
        GtkClipboard *clipboard;
+       EvLinkAction *ev_action;
        const gchar *uri;
 
-       uri = ev_link_get_uri (window->priv->link);
+       ev_action = ev_link_get_action (window->priv->link);
+       if (!ev_action)
+               return;
+
+       uri = ev_link_action_get_uri (ev_action);
 
        clipboard = gtk_widget_get_clipboard (GTK_WIDGET (window),
                                              GDK_SELECTION_CLIPBOARD);
@@ -3301,18 +3358,25 @@ ev_view_popup_cmd_copy_link_address (GtkAction *action, EvWindow *window)
 static void
 ev_attachment_popup_cmd_open_attachment (GtkAction *action, EvWindow *window)
 {
-       GError *error = NULL;
+       GList *l;
        
-       if (!window->priv->attachment)
+       if (!window->priv->attach_list)
                return;
-       
-       ev_attachment_open (window->priv->attachment, &error);
 
-       if (error) {
-               ev_window_error_dialog (GTK_WINDOW (window),
-                                       _("Unable to open attachment"),
-                                       error);
-               g_error_free (error);
+       for (l = window->priv->attach_list; l && l->data; l = g_list_next (l)) {
+               EvAttachment *attachment;
+               GError       *error = NULL;
+               
+               attachment = (EvAttachment *) l->data;
+               
+               ev_attachment_open (attachment, &error);
+
+               if (error) {
+                       ev_window_error_dialog (GTK_WINDOW (window),
+                                               _("Unable to open attachment"),
+                                               error);
+                       g_error_free (error);
+               }
        }
 }
 
@@ -3321,37 +3385,72 @@ attachment_save_dialog_response_cb (GtkWidget *fc,
                                    gint       response_id,
                                    EvWindow  *ev_window)
 {
-       if (response_id == GTK_RESPONSE_OK) {
-               const char *uri;
-               GError     *error = NULL;
-
-               uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (fc));
+       gchar                *uri;
+       GList                *l;
+       GtkFileChooserAction  fc_action;
+       gboolean              is_dir;
+       
+       if (response_id != GTK_RESPONSE_OK) {
+               gtk_widget_destroy (fc);
+               return;
+       }
 
-               ev_attachment_save (ev_window->priv->attachment, uri, &error);
+       uri = gtk_file_chooser_get_uri (GTK_FILE_CHOOSER (fc));
+       
+       g_object_get (G_OBJECT (fc), "action", &fc_action, NULL);
+       is_dir = (fc_action == GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER);
+       
+       for (l = ev_window->priv->attach_list; l && l->data; l = g_list_next (l)) {
+               EvAttachment *attachment;
+               gchar        *filename;
+               GError       *error = NULL;
+               
+               attachment = (EvAttachment *) l->data;
 
+               if (is_dir) {
+                       filename = g_strjoin ("/", uri,
+                                             ev_attachment_get_name (attachment),
+                                             NULL);
+               } else {
+                       filename = g_strdup (uri);
+               }
+               
+               ev_attachment_save (attachment, filename, &error);
+               g_free (filename);
+               
                if (error) {
                        ev_window_error_dialog (GTK_WINDOW (fc),
-                                                _("The attachment could not be saved."),
-                                                error);
+                                               _("The attachment could not be saved."),
+                                               error);
                        g_error_free (error);
                }
        }
 
+       g_free (uri);
+
        gtk_widget_destroy (fc);
 }
 
 static void
 ev_attachment_popup_cmd_save_attachment_as (GtkAction *action, EvWindow *window)
 {
-       GtkWidget *fc;
+       GtkWidget    *fc;
+       EvAttachment *attachment = NULL;
 
-       fc = gtk_file_chooser_dialog_new (_("Save a Copy"),
-                                         GTK_WINDOW (window),
-                                         GTK_FILE_CHOOSER_ACTION_SAVE,
-                                         GTK_STOCK_CANCEL,
-                                         GTK_RESPONSE_CANCEL,
-                                         GTK_STOCK_SAVE, GTK_RESPONSE_OK,
-                                         NULL);
+       if (!window->priv->attach_list)
+               return;
+
+       if (g_list_length (window->priv->attach_list) == 1)
+               attachment = (EvAttachment *) window->priv->attach_list->data;
+       
+       fc = gtk_file_chooser_dialog_new (
+               _("Save a Copy"),
+               GTK_WINDOW (window),
+               attachment ? GTK_FILE_CHOOSER_ACTION_SAVE : GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
+               GTK_STOCK_CANCEL,
+               GTK_RESPONSE_CANCEL,
+               GTK_STOCK_SAVE, GTK_RESPONSE_OK,
+               NULL);
 
        gtk_dialog_set_default_response (GTK_DIALOG (fc), GTK_RESPONSE_OK);
 
@@ -3359,8 +3458,9 @@ ev_attachment_popup_cmd_save_attachment_as (GtkAction *action, EvWindow *window)
        gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (fc), TRUE);
 #endif
 
-       gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (fc),
-                                          ev_attachment_get_name (window->priv->attachment));
+       if (attachment)
+               gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (fc),
+                                                  ev_attachment_get_name (attachment));
 
        g_signal_connect (fc, "response",
                          G_CALLBACK (attachment_save_dialog_response_cb),
@@ -3384,6 +3484,8 @@ ev_window_init (EvWindow *ev_window)
 
        ev_window->priv = EV_WINDOW_GET_PRIVATE (ev_window);
 
+       ev_window->priv->dest = NULL;
+
        ev_window->priv->page_mode = PAGE_MODE_DOCUMENT;
        ev_window->priv->title = ev_window_title_new (ev_window);
 
@@ -3479,6 +3581,27 @@ ev_window_init (EvWindow *ev_window)
        gtk_widget_show (ev_window->priv->sidebar);
 
        /* Stub sidebar, for now */
+
+       sidebar_widget = ev_sidebar_thumbnails_new ();
+       ev_window->priv->sidebar_thumbs = sidebar_widget;
+       g_signal_connect (sidebar_widget,
+                         "notify::main-widget",
+                         G_CALLBACK (sidebar_page_main_widget_update_cb),
+                         ev_window);
+       sidebar_page_main_widget_update_cb (G_OBJECT (sidebar_widget), NULL, ev_window);
+       gtk_widget_show (sidebar_widget);
+       ev_sidebar_add_page (EV_SIDEBAR (ev_window->priv->sidebar),
+                            sidebar_widget);
+
+       sidebar_widget = ev_sidebar_attachments_new ();
+       ev_window->priv->sidebar_attachments = sidebar_widget;
+       g_signal_connect_object (sidebar_widget,
+                                "popup",
+                                G_CALLBACK (attachment_bar_menu_popup_cb),
+                                ev_window, 0);
+       ev_sidebar_add_page (EV_SIDEBAR (ev_window->priv->sidebar),
+                            sidebar_widget);
+
        sidebar_widget = ev_sidebar_links_new ();
        ev_window->priv->sidebar_links = sidebar_widget;
        g_signal_connect (sidebar_widget,
@@ -3494,18 +3617,6 @@ ev_window_init (EvWindow *ev_window)
        ev_sidebar_add_page (EV_SIDEBAR (ev_window->priv->sidebar),
                             sidebar_widget);
 
-       sidebar_widget = ev_sidebar_thumbnails_new ();
-       ev_window->priv->sidebar_thumbs = sidebar_widget;
-       g_signal_connect (sidebar_widget,
-                         "notify::main-widget",
-                         G_CALLBACK (sidebar_page_main_widget_update_cb),
-                         ev_window);
-       sidebar_page_main_widget_update_cb (G_OBJECT (sidebar_widget), NULL, ev_window);
-       gtk_widget_show (sidebar_widget);
-       ev_sidebar_add_page (EV_SIDEBAR (ev_window->priv->sidebar),
-                            sidebar_widget);
-
-
        ev_window->priv->scrolled_window =
                GTK_WIDGET (g_object_new (GTK_TYPE_SCROLLED_WINDOW,
                                          "shadow-type", GTK_SHADOW_IN,
@@ -3537,16 +3648,6 @@ ev_window_init (EvWindow *ev_window)
        gtk_widget_show (ev_window->priv->view);
        gtk_widget_show (ev_window->priv->password_view);
 
-       /* Attachments Bar */
-       ev_window->priv->attachment_bar = ev_attachment_bar_new ();
-       gtk_box_pack_end (GTK_BOX (ev_window->priv->main_box),
-                         ev_window->priv->attachment_bar,
-                         FALSE, TRUE, 0);
-       g_signal_connect_object (ev_window->priv->attachment_bar,
-                                "popup",
-                                G_CALLBACK (attachment_bar_menu_popup_cb),
-                                ev_window, 0);
-
        /* Find Bar */
        ev_window->priv->find_bar = egg_find_bar_new ();
        gtk_box_pack_end (GTK_BOX (ev_window->priv->main_box),
@@ -3599,12 +3700,6 @@ ev_window_init (EvWindow *ev_window)
                          G_CALLBACK (ev_window_sidebar_current_page_changed_cb),
                          ev_window);
 
-       /* Connect attachment bar sgignals */
-       g_signal_connect (G_OBJECT (ev_window->priv->attachment_bar),
-                         "notify::expanded",
-                         G_CALLBACK (ev_window_attachment_bar_toggled_cb),
-                         ev_window);
-
        /* Connect to find bar signals */
        g_signal_connect (ev_window->priv->find_bar,
                          "previous",
@@ -3642,31 +3737,31 @@ ev_window_init (EvWindow *ev_window)
 
        ev_window->priv->attachment_popup = gtk_ui_manager_get_widget (ev_window->priv->ui_manager,
                                                                       "/AttachmentPopup");
-       ev_window->priv->attachment = NULL;
+       ev_window->priv->attach_list = NULL;
 
        /* Give focus to the document view */
        gtk_widget_grab_focus (ev_window->priv->view);
 
        /* Drag and Drop */
-       gtk_drag_dest_unset (GTK_WIDGET (ev_window));
-       gtk_drag_dest_set (GTK_WIDGET (ev_window), GTK_DEST_DEFAULT_ALL,
+       gtk_drag_dest_unset (GTK_WIDGET (ev_window->priv->view));
+       gtk_drag_dest_set (GTK_WIDGET (ev_window->priv->view),
+                          GTK_DEST_DEFAULT_ALL,
                           ev_drop_types,
                           sizeof (ev_drop_types) / sizeof (ev_drop_types[0]),
                           GDK_ACTION_COPY);
-       g_signal_connect (G_OBJECT (ev_window), "drag-data-received",
-                         G_CALLBACK (drag_data_received_cb), NULL);
-
+       g_signal_connect_swapped (G_OBJECT (ev_window->priv->view), "drag-data-received",
+                                 G_CALLBACK (drag_data_received_cb),
+                                 ev_window);
+       
        /* Set it user interface params */
 
        ev_window_setup_recent (ev_window);
+
        setup_chrome_from_metadata (ev_window);
        set_chrome_actions (ev_window);
        update_chrome_visibility (ev_window);
-       update_sidebar_visibility (ev_window);
-       update_attachment_bar_visibility (ev_window);
 
-       gtk_window_set_default_size (GTK_WINDOW (ev_window),
-                                    600, 600);
+       gtk_window_set_default_size (GTK_WINDOW (ev_window), 600, 600);
        setup_view_from_metadata (ev_window);
 
         ev_window_sizing_mode_changed_cb (EV_VIEW (ev_window->priv->view), NULL, ev_window);