Set the proper screen size of ModestShell window
[modest] / src / gtk / modest-shell.c
index 1e432c0..a9a296e 100644 (file)
 #include <string.h>
 #include <modest-shell.h>
 #include <modest-shell-window.h>
+#include <modest-icon-names.h>
+#include <modest-ui-actions.h>
+#include <modest-gtk-window-mgr.h>
+#include <modest-runtime.h>
+#include <X11/Xutil.h>
+
+#define SCN_WIDTH  1024
+#define SCN_HEIGHT 600
 
 /* 'private'/'protected' functions */
 static void modest_shell_class_init (ModestShellClass *klass);
 static void modest_shell_instance_init (ModestShell *obj);
+static void modest_shell_dispose   (GObject *obj);
 static void modest_shell_finalize   (GObject *obj);
 
 static void update_title (ModestShell *self);
 
 static void on_back_button_clicked (GtkToolButton *button, ModestShell *self);
 static void on_title_button_clicked (GtkToolButton *button, ModestShell *self);
+static void on_new_msg_button_clicked (GtkToolButton *button, ModestShell *self);
+static void on_style_set (GtkWidget *widget, GtkStyle *old_style, ModestShell *shell);
+static gboolean on_delete_event (GtkWidget *widget,
+                                GdkEvent  *event,
+                                gpointer   user_data);
+static gboolean on_control_q (GtkAccelGroup *accel_group,
+                             GObject *acceleratable,
+                             guint keyval,
+                             GdkModifierType modifier);
+static gboolean on_key_pressed (GtkWidget *widget, GdkEventKey *event, ModestShell *shell);
 
 
 typedef struct _ModestShellPrivate ModestShellPrivate;
@@ -47,10 +66,18 @@ struct _ModestShellPrivate {
        GtkWidget *main_vbox;
        GtkWidget *notebook;
        GtkWidget *top_toolbar;
+       GtkToolItem *new_message_button;
        GtkToolItem *back_button;
        GtkToolItem *title_button;
        GtkWidget *title_label;
        GtkWidget *subtitle_label;
+
+       GtkWidget *progress_icon;
+       GdkPixbuf **progress_frames;
+       gint next_frame;
+       guint progress_timeout_id;
+
+       GtkWidget *banners_box;
 };
 #define MODEST_SHELL_GET_PRIVATE(o)      (G_TYPE_INSTANCE_GET_PRIVATE((o), \
                                                                      MODEST_TYPE_SHELL, \
@@ -90,6 +117,7 @@ modest_shell_class_init (ModestShellClass *klass)
        gobject_class = (GObjectClass*) klass;
 
        parent_class            = g_type_class_peek_parent (klass);
+       gobject_class->dispose = modest_shell_dispose;
        gobject_class->finalize = modest_shell_finalize;
 
        g_type_class_add_private (gobject_class, sizeof(ModestShellPrivate));
@@ -97,30 +125,84 @@ modest_shell_class_init (ModestShellClass *klass)
 }
 
 static void
+set_screen_size (GtkWindow *self)
+{
+       /* Get screen dimensions */
+       Screen *screen = XDefaultScreenOfDisplay(XOpenDisplay(NULL));
+       int scn_width = XWidthOfScreen(screen);
+       int scn_height = XHeightOfScreen(screen);
+
+       /* if screensize is larger than netbook size, show in window, else fullscreen */
+       if (scn_width > SCN_WIDTH || scn_height > SCN_HEIGHT) {
+               scn_width = SCN_WIDTH;
+               scn_height = SCN_HEIGHT;
+               gtk_window_resize (self, scn_width, scn_height);
+       } else {
+               gtk_window_fullscreen (self);
+               /* Very important for Moblin */
+               gtk_window_set_decorated (self, FALSE);
+       }
+}
+
+static void
 modest_shell_instance_init (ModestShell *obj)
 {
        ModestShellPrivate *priv;
        GtkWidget *title_vbox;
+       GtkWidget *title_arrow;
+       GtkWidget *new_message_icon;
+       GtkToolItem *separator_toolitem;
+       GtkWidget *top_hbox;
+       GtkWidget *separator;
 
        priv = MODEST_SHELL_GET_PRIVATE(obj);
+       priv->progress_frames = g_malloc0 (sizeof(GdkPixbuf *)*31);
+       priv->progress_timeout_id = 0;
+       priv->next_frame = 0;
 
        priv->main_vbox = gtk_vbox_new (FALSE, 0);
        gtk_widget_show (priv->main_vbox);
 
+       top_hbox = gtk_hbox_new (FALSE, 0);
+       gtk_widget_show (top_hbox);
+       gtk_box_pack_start (GTK_BOX (priv->main_vbox), top_hbox, FALSE, FALSE, 0);
+
+       separator = gtk_hseparator_new ();
+       gtk_widget_show (separator);
+       gtk_box_pack_start (GTK_BOX (priv->main_vbox), separator, FALSE, FALSE, 0);
+
        priv->top_toolbar = gtk_toolbar_new ();
        gtk_toolbar_set_style (GTK_TOOLBAR (priv->top_toolbar), GTK_TOOLBAR_BOTH_HORIZ);
+       gtk_toolbar_set_show_arrow (GTK_TOOLBAR (priv->top_toolbar), FALSE);
        gtk_widget_show (priv->top_toolbar);
-       gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->top_toolbar, FALSE, FALSE, 0);
+       gtk_box_pack_start (GTK_BOX (top_hbox), priv->top_toolbar, TRUE, TRUE, 0);
 
-       priv->back_button = gtk_tool_button_new_from_stock (GTK_STOCK_GO_BACK);
-       gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->back_button, -1);
-       gtk_widget_show (GTK_WIDGET (priv->back_button));
-       g_signal_connect (G_OBJECT (priv->back_button), "clicked", G_CALLBACK (on_back_button_clicked), obj);
+       priv->progress_icon = gtk_image_new ();
+       gtk_widget_show (priv->progress_icon);
+       gtk_box_pack_start (GTK_BOX (top_hbox), priv->progress_icon, FALSE, FALSE, 0);
+
+       priv->banners_box = gtk_vbox_new (FALSE, 0);
+       gtk_widget_show (priv->banners_box);
+       gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->banners_box, FALSE, FALSE, 0);
+
+       new_message_icon = gtk_image_new_from_icon_name (MODEST_TOOLBAR_ICON_NEW_MAIL, GTK_ICON_SIZE_LARGE_TOOLBAR);
+       gtk_widget_show (new_message_icon);
+       priv->new_message_button = gtk_tool_button_new (new_message_icon, _("mcen_va_new_email"));
+       g_object_set (priv->new_message_button, "is-important", TRUE, NULL);
+       gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->new_message_button, -1);
+       gtk_widget_show (GTK_WIDGET (priv->new_message_button));
+       g_signal_connect (G_OBJECT (priv->new_message_button), "clicked", G_CALLBACK (on_new_msg_button_clicked), obj);
+
+       separator_toolitem = gtk_separator_tool_item_new ();
+       gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), separator_toolitem, -1);
+       gtk_widget_show (GTK_WIDGET (separator_toolitem));
 
        title_vbox = gtk_vbox_new (FALSE, 0);
        priv->title_label = gtk_label_new (NULL);
+       gtk_label_set_ellipsize (GTK_LABEL (priv->title_label), PANGO_ELLIPSIZE_END);
        gtk_misc_set_alignment (GTK_MISC (priv->title_label), 0.0, 1.0);
        priv->subtitle_label = gtk_label_new (NULL);
+       gtk_label_set_ellipsize (GTK_LABEL (priv->subtitle_label), PANGO_ELLIPSIZE_START);
        gtk_misc_set_alignment (GTK_MISC (priv->subtitle_label), 0.0, 0.0);
        gtk_widget_show (priv->title_label);
        gtk_widget_show (priv->subtitle_label);
@@ -128,24 +210,93 @@ modest_shell_instance_init (ModestShell *obj)
        gtk_box_pack_start (GTK_BOX (title_vbox), priv->subtitle_label, FALSE, FALSE, 0);
        gtk_widget_show (title_vbox);
 
-       priv->title_button = gtk_tool_button_new (NULL, NULL);
+       priv->title_button = gtk_tool_button_new (NULL, "");
        gtk_widget_show (GTK_WIDGET (priv->title_button));
+       title_arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE);
+       gtk_widget_show (title_arrow);
+       gtk_tool_button_set_icon_widget (GTK_TOOL_BUTTON (priv->title_button), title_arrow);
        gtk_tool_button_set_label_widget (GTK_TOOL_BUTTON (priv->title_button), title_vbox);
        gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->title_button, -1);
        gtk_container_child_set (GTK_CONTAINER (priv->top_toolbar), GTK_WIDGET (priv->title_button), "expand", TRUE, NULL);
        g_object_set (priv->title_button, "is-important", TRUE, NULL);
        g_signal_connect (G_OBJECT (priv->title_button), "clicked", G_CALLBACK (on_title_button_clicked), obj);
 
+       priv->back_button = gtk_tool_button_new_from_stock (GTK_STOCK_GO_BACK);
+       g_object_set (priv->back_button, "is-important", TRUE, NULL);
+       gtk_toolbar_insert (GTK_TOOLBAR (priv->top_toolbar), priv->back_button, -1);
+       gtk_widget_show (GTK_WIDGET (priv->back_button));
+       g_signal_connect (G_OBJECT (priv->back_button), "clicked", G_CALLBACK (on_back_button_clicked), obj);
+
        priv->notebook = gtk_notebook_new ();
+       gtk_notebook_set_show_tabs ((GtkNotebook *)priv->notebook, FALSE);
+       gtk_notebook_set_show_border ((GtkNotebook *)priv->notebook, FALSE);
        gtk_widget_show (priv->notebook);
        gtk_box_pack_start (GTK_BOX (priv->main_vbox), priv->notebook, TRUE, TRUE, 0);
        gtk_container_add (GTK_CONTAINER (obj), priv->main_vbox);
 
+       g_signal_connect (G_OBJECT (obj), "style-set", G_CALLBACK (on_style_set), obj);
+       g_signal_connect (G_OBJECT (obj), "delete-event", G_CALLBACK (on_delete_event), obj);
+
+       guint accel_key;
+       GdkModifierType accel_mods;
+       GtkAccelGroup *accel_group;
+       accel_group = gtk_accel_group_new ();
+       gtk_accelerator_parse ("<Control>n", &accel_key, &accel_mods);
+       gtk_widget_add_accelerator (GTK_WIDGET (priv->new_message_button), "clicked", accel_group,
+                                   accel_key, accel_mods, 0);
+       gtk_accelerator_parse ("Escape", &accel_key, &accel_mods);
+       gtk_widget_add_accelerator (GTK_WIDGET (priv->back_button), "clicked", accel_group,
+                                   accel_key, accel_mods, 0);
+       gtk_accelerator_parse ("<Control>w", &accel_key, &accel_mods);
+       gtk_widget_add_accelerator (GTK_WIDGET (priv->back_button), "clicked", accel_group,
+                                   accel_key, accel_mods, 0);
+       gtk_accelerator_parse ("<Control>q", &accel_key, &accel_mods);
+       gtk_accel_group_connect (accel_group, accel_key, accel_mods, 0, g_cclosure_new (G_CALLBACK (on_control_q), NULL, NULL));
+       gtk_accelerator_parse ("F10", &accel_key, &accel_mods);
+       gtk_widget_add_accelerator (GTK_WIDGET (priv->title_button), "clicked", accel_group,
+                                   accel_key, accel_mods, 0);
+       gtk_window_add_accel_group (GTK_WINDOW (obj), accel_group);
+
+       g_signal_connect (G_OBJECT (obj), 
+                         "key-press-event", 
+                         G_CALLBACK (on_key_pressed), obj);
+
+       set_screen_size ((GtkWindow *) obj);
+}
+
+static void
+modest_shell_dispose (GObject *obj)
+{
+       ModestShellPrivate *priv;
+
+       priv = MODEST_SHELL_GET_PRIVATE (obj);
+
+       if (priv->progress_timeout_id) {
+               g_source_remove (priv->progress_timeout_id);
+               priv->progress_timeout_id = 0;
+       }
+       G_OBJECT_CLASS(parent_class)->dispose (obj);
 }
 
 static void
 modest_shell_finalize (GObject *obj)
 {
+       ModestShellPrivate *priv;
+       int n;
+
+       priv = MODEST_SHELL_GET_PRIVATE (obj);
+
+       if (priv->progress_timeout_id) {
+               g_source_remove (priv->progress_timeout_id);
+               priv->progress_timeout_id = 0;
+       }
+       for (n = 0; n < 31; n++) {
+               if (priv->progress_frames[n]) {
+                       g_object_unref (priv->progress_frames[n]);
+               }
+       }
+       g_free (priv->progress_frames);
+
        G_OBJECT_CLASS(parent_class)->finalize (obj);
 }
 
@@ -180,12 +331,13 @@ modest_shell_delete_window (ModestShell *shell, ModestWindow *window)
        priv = MODEST_SHELL_GET_PRIVATE (shell);
        g_signal_emit_by_name (G_OBJECT (window), "delete-event", NULL, &ret_value);
        if (ret_value == FALSE) {
-               gint page_num;
-               
-               page_num = gtk_notebook_page_num (GTK_NOTEBOOK (priv->notebook), GTK_WIDGET (window));
-               if (page_num != -1) {
-                       gtk_notebook_remove_page (GTK_NOTEBOOK (priv->notebook), page_num);
+               GList *children;
+
+               children = gtk_container_get_children (GTK_CONTAINER (priv->notebook));
+               if (g_list_find (children, window)) {
+                       gtk_container_remove (GTK_CONTAINER (priv->notebook), GTK_WIDGET (window));
                }
+               
        }
 
        update_title (shell);
@@ -228,9 +380,47 @@ modest_shell_set_title (ModestShell *shell, ModestWindow *window, const gchar *t
        update_title (shell);
 }
 
+static void
+show_next_frame (ModestShell *shell)
+{
+       ModestShellPrivate *priv;
+
+       priv = MODEST_SHELL_GET_PRIVATE (shell);
+
+       gtk_image_set_from_pixbuf (GTK_IMAGE (priv->progress_icon), priv->progress_frames[priv->next_frame]);
+
+       priv->next_frame++;
+       if (priv->next_frame >= 31)
+               priv->next_frame = 0;
+}
+
+static gboolean
+on_progress_timeout (ModestShell *shell)
+{
+       show_next_frame (shell);
+       return TRUE;
+}
+
 void
 modest_shell_show_progress (ModestShell *shell, ModestWindow *window, gboolean show)
 {
+       ModestShellPrivate *priv;
+
+       priv = MODEST_SHELL_GET_PRIVATE (shell);
+
+       if (show) {
+               if (priv->progress_timeout_id == 0) {
+                       priv->progress_timeout_id = g_timeout_add (100, (GSourceFunc) on_progress_timeout, shell);
+                       show_next_frame (shell);
+               }
+               gtk_widget_show (priv->progress_icon);
+       } else {
+               if (priv->progress_timeout_id) {
+                       g_source_remove (priv->progress_timeout_id);
+                       priv->progress_timeout_id = 0;
+               }
+               gtk_widget_hide (priv->progress_icon);
+       }
 }
 
 static void
@@ -241,6 +431,7 @@ update_title (ModestShell *self)
        GtkWidget *child;
        GString *title_buffer;
        GString *subtitle_buffer;
+       const gchar *tab_label_text;
 
        priv = MODEST_SHELL_GET_PRIVATE (self);
 
@@ -254,7 +445,9 @@ update_title (ModestShell *self)
        child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), n_pages - 1);
        title_buffer = g_string_new ("");
        title_buffer = g_string_append (title_buffer, "<b>");
-       title_buffer = g_string_append (title_buffer, gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child));
+       tab_label_text = gtk_notebook_get_tab_label_text (GTK_NOTEBOOK (priv->notebook), child);
+       if (tab_label_text)
+               title_buffer = g_string_append (title_buffer, tab_label_text);
        title_buffer = g_string_append (title_buffer, "</b>");
        gtk_label_set_markup (GTK_LABEL (priv->title_label), 
                              title_buffer->str);
@@ -299,11 +492,33 @@ on_back_button_clicked (GtkToolButton *button, ModestShell *self)
 }
 
 static void
+menu_position_cb (GtkMenu *menu,
+                 gint *x,
+                 gint *y,
+                 gboolean *push_in,
+                 ModestShell *self)
+{
+       ModestShellPrivate *priv;
+       GtkAllocation *alloc;
+       GdkWindow *parent_window;
+       gint pos_x, pos_y;
+
+       priv = MODEST_SHELL_GET_PRIVATE (self);
+
+       alloc = &(GTK_WIDGET (priv->title_button)->allocation);
+       parent_window = gtk_widget_get_parent_window (GTK_WIDGET (priv->title_button));
+       gdk_window_get_position (parent_window, &pos_x, &pos_y);
+       *x = pos_x + alloc->x;
+       *y = pos_y + alloc->y + alloc->height;
+       *push_in = TRUE;
+       
+}
+
+static void
 on_title_button_clicked (GtkToolButton *button, ModestShell *self)
 {
        ModestShellPrivate *priv;
        gint n_pages;
-       gboolean delete_event_retval;
        GtkWidget *child;
        GtkWidget *menu;
 
@@ -317,6 +532,140 @@ on_title_button_clicked (GtkToolButton *button, ModestShell *self)
        menu = modest_shell_window_get_menu (MODEST_SHELL_WINDOW (child));
 
        if (menu) {
-               gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL, 1, gtk_get_current_event_time ());
+               /* Run dimming rules */
+               ModestWindow *window = modest_shell_peek_window (self);
+               modest_ui_actions_check_menu_dimming_rules (window);
+
+               gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
+                               (GtkMenuPositionFunc) menu_position_cb, (gpointer) self,
+                               1, gtk_get_current_event_time ());
+       }
+}
+
+static void
+on_new_msg_button_clicked (GtkToolButton *button, ModestShell *self)
+{
+       ModestShellPrivate *priv;
+       gint n_pages;
+       GtkWidget *child;
+
+       priv = MODEST_SHELL_GET_PRIVATE (self);
+
+       n_pages = gtk_notebook_get_n_pages (GTK_NOTEBOOK (priv->notebook));
+       if (n_pages < 1)
+               return;
+
+       child = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
+
+       modest_ui_actions_on_new_msg (NULL, MODEST_WINDOW (child));
+}
+
+static void
+on_style_set (GtkWidget *widget,
+             GtkStyle *old_style,
+             ModestShell *self)
+{
+       ModestShellPrivate *priv;
+       gint icon_w, icon_h;
+       GdkPixbuf *progress_pixbuf;
+       int n;
+
+       priv = MODEST_SHELL_GET_PRIVATE (self);
+
+       if (!gtk_icon_size_lookup (GTK_ICON_SIZE_LARGE_TOOLBAR, &icon_w, &icon_h))
+               return;
+       progress_pixbuf = gtk_icon_theme_load_icon (gtk_icon_theme_get_default (), "process-working", icon_w, 0, NULL);
+
+       for (n = 0; n < 31; n++) {
+               if (priv->progress_frames[n] != NULL) {
+                       g_object_unref (priv->progress_frames[n]);
+               }
+               priv->progress_frames[n] = NULL;
        }
+
+       if (progress_pixbuf) {
+               gint max_x, max_y;
+               gint i, j;
+
+               icon_w = gdk_pixbuf_get_width (progress_pixbuf) / 8;
+
+               n = 0;
+               max_x = 8;
+               max_y = 4;
+               for (i = 0; i < 4; i++) {
+                       for (j = 0; j < 8; j++) {
+                                       GdkPixbuf *frame;
+
+                                       if ((i == 0) && (j == 0))
+                                               continue;
+                                       frame = gdk_pixbuf_new_subpixbuf  (progress_pixbuf,
+                                                                          j*icon_w, i*icon_w,
+                                                                          icon_w, icon_w);
+                                       priv->progress_frames[n] = frame;
+                                       n++;
+                               }
+                       }
+               g_object_unref (progress_pixbuf);
+       }
+
+}
+
+static gboolean
+on_key_pressed (GtkWidget *widget,
+               GdkEventKey *event,
+               ModestShell *shell)
+{
+       ModestShellPrivate *priv;
+       gboolean retval;
+       GtkWidget *current_window;
+
+       priv = MODEST_SHELL_GET_PRIVATE (shell);
+
+       current_window = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->notebook), -1);
+
+       g_signal_emit_by_name (current_window, "key-press-event", event, &retval);
+
+       return retval;
+       
+}
+
+void
+modest_shell_add_banner (ModestShell *shell, ModestShellBanner *banner)
+{
+       ModestShellPrivate *priv;
+
+       priv = MODEST_SHELL_GET_PRIVATE (shell);
+       gtk_box_pack_start (GTK_BOX (priv->banners_box), GTK_WIDGET (banner), FALSE, FALSE, 0);
+}
+
+static gboolean 
+on_delete_event (GtkWidget *widget,
+                GdkEvent  *event,
+                gpointer   user_data)
+{
+       ModestWindow *top_window;
+
+       while ((top_window = modest_shell_peek_window (MODEST_SHELL (widget))) != NULL) {
+               if (modest_shell_delete_window (MODEST_SHELL (widget), top_window))
+                       return TRUE;
+       }
+
+       return FALSE;
+}
+
+static gboolean 
+on_control_q (GtkAccelGroup *accel_group,
+             GObject *acceleratable,
+             guint keyval,
+             GdkModifierType modifier)
+{
+       ModestWindowMgr *mgr;
+       GtkWidget *shell;
+       gboolean ret_value;
+
+       mgr = modest_runtime_get_window_mgr ();
+       shell = modest_gtk_window_mgr_get_shell (MODEST_GTK_WINDOW_MGR (mgr));
+       g_signal_emit_by_name (G_OBJECT (shell), "delete-event", NULL, &ret_value);
+
+       return ret_value;
 }