aboutsummaryrefslogtreecommitdiffstats
path: root/shell/e-shell-folder-title-bar.c
diff options
context:
space:
mode:
Diffstat (limited to 'shell/e-shell-folder-title-bar.c')
-rw-r--r--shell/e-shell-folder-title-bar.c266
1 files changed, 102 insertions, 164 deletions
diff --git a/shell/e-shell-folder-title-bar.c b/shell/e-shell-folder-title-bar.c
index 7e90e9bbce..f1fef8c68b 100644
--- a/shell/e-shell-folder-title-bar.c
+++ b/shell/e-shell-folder-title-bar.c
@@ -46,27 +46,30 @@ static GtkHBox *parent_class = NULL;
struct _EShellFolderTitleBarPrivate {
GdkPixbuf *icon;
- GtkWidget *icon_widget;
- /* We have a label and a button. When the button is enabled,
- the label is hidden; when the button is disable, only the
- label is visible. */
+ /* We have an icon, a label and a button that contains an icon and a
+ label. When the button is enabled, the stand-alone label icon get
+ hidden; when the button is disabled, the button gets hidden and the
+ label and the icon get shown. This is pretty ugly but it easier to
+ manage the GTK layout this way. */
- /* The label. */
+ /* The stand-alone icon/label combo. */
+ GtkWidget *title_icon;
GtkWidget *title_label;
- /* Holds extra information that is to be shown to the left of the icon */
+ /* The button. */
+ GtkWidget *title_button;
+ GtkWidget *title_button_icon;
+ GtkWidget *title_button_label;
+ GtkWidget *title_button_arrow;
+
+ /* Holds extra information that is to be shown on the bar. */
GtkWidget *folder_bar_label;
/* Navigation buttons. */
GtkWidget *back_button;
GtkWidget *forward_button;
- /* The button. */
- GtkWidget *title_button;
- GtkWidget *title_button_label;
- GtkWidget *title_button_arrow;
-
gboolean title_clickable;
};
@@ -118,7 +121,7 @@ static const char *right_arrow_xpm[] = {
};
-/* Icon pixmap. */
+/* Utility functions for managing icons and icon widgets. */
static GtkWidget *
create_pixmap_widget_from_xpm (const char **xpm)
@@ -138,53 +141,39 @@ create_pixmap_widget_from_xpm (const char **xpm)
return widget;
}
-
-#if 0 /* This code is kinda broken in some subtle way
- I haven't been able to figure out. */
-
-static void
-label_realize_callback (GtkWidget *widget,
- void *data)
+static GdkPixbuf *
+new_empty_pixbuf (void)
{
- GtkStyle *style;
- EFont *e_font;
- GdkFont *bolded_font;
+ GdkPixbuf *empty_pixbuf;
+ unsigned char *pixels;
- g_assert (widget->style->font != NULL);
+ empty_pixbuf = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8, 1, 1);
+ pixels = gdk_pixbuf_get_pixels (empty_pixbuf);
- style = gtk_style_copy (widget->style);
- gtk_style_unref (widget->style);
- widget->style = style;
+ memset (pixels, 0, 4);
- e_font = e_font_from_gdk_font (style->font);
- bolded_font = e_font_to_gdk_font (e_font, E_FONT_BOLD);
- e_font_unref (e_font);
+ return empty_pixbuf;
+}
- if (bolded_font != NULL) {
- gdk_font_unref (style->font);
- style->font = bolded_font;
- }
+static GtkWidget *
+new_empty_pixmap_widget (void)
+{
+ GtkWidget *pixmap_widget;
+ GdkPixmap *pixmap;
+ GdkBitmap *mask;
+ GdkPixbuf *empty_pixbuf;
- gtk_style_attach (style, widget->window);
+ empty_pixbuf = new_empty_pixbuf ();
- if (E_IS_CLIPPED_LABEL (widget)) {
- char *text;
+ gdk_pixbuf_render_pixmap_and_mask (empty_pixbuf, &pixmap, &mask, 127);
+ pixmap_widget = gtk_pixmap_new (pixmap, mask);
- text = g_strdup (e_clipped_label_get_text (E_CLIPPED_LABEL (widget)));
- e_clipped_label_set_text (E_CLIPPED_LABEL (widget), text);
- g_free (text);
- }
-}
+ gdk_pixbuf_unref (empty_pixbuf);
-static void
-make_bold (GtkWidget *widget)
-{
- gtk_signal_connect (GTK_OBJECT (widget), "realize",
- GTK_SIGNAL_FUNC (label_realize_callback), NULL);
+ return pixmap_widget;
}
-#endif
-
+
static void
set_title_bar_label_style (GtkWidget *widget)
{
@@ -219,35 +208,6 @@ get_max_clipped_label_width (EClippedLabel *clipped_label)
}
static void
-size_allocate_icon (EShellFolderTitleBar *title_bar,
- GtkAllocation *allocation,
- int *available_width_inout)
-{
- EShellFolderTitleBarPrivate *priv;
- GtkRequisition icon_requisition;
- GtkAllocation icon_allocation;
- int border_width;
-
- priv = title_bar->priv;
-
- if (priv->icon_widget == NULL)
- return;
-
- border_width = GTK_CONTAINER (title_bar)->border_width;
-
- gtk_widget_get_child_requisition (priv->icon_widget, &icon_requisition);
-
- icon_allocation.x = allocation->x + allocation->width - border_width - icon_requisition.width;
- icon_allocation.y = allocation->y + border_width;
- icon_allocation.width = icon_requisition.width;
- icon_allocation.height = allocation->height - 2 * border_width;
-
- gtk_widget_size_allocate (priv->icon_widget, &icon_allocation);
-
- *available_width_inout -= icon_allocation.width;
-}
-
-static void
size_allocate_title_button (EShellFolderTitleBar *title_bar,
GtkAllocation *allocation,
int offset,
@@ -260,6 +220,9 @@ size_allocate_title_button (EShellFolderTitleBar *title_bar,
priv = title_bar->priv;
+ /* Keep a little distance from the navigation arrows. */
+ allocation->x += 2;
+
border_width = GTK_CONTAINER (title_bar)->border_width;
gtk_widget_get_child_requisition (priv->title_button, &child_requisition);
@@ -278,8 +241,8 @@ size_allocate_title_button (EShellFolderTitleBar *title_bar,
}
static int
-size_allocate_navigation_buttons (EShellFolderTitleBar *title_bar,
- GtkAllocation *allocation)
+size_allocate_navigation_buttons_and_title_icon (EShellFolderTitleBar *title_bar,
+ GtkAllocation *allocation)
{
EShellFolderTitleBarPrivate *priv;
GtkRequisition child_requisition;
@@ -304,6 +267,15 @@ size_allocate_navigation_buttons (EShellFolderTitleBar *title_bar,
child_allocation.width = child_requisition.width;
gtk_widget_size_allocate (priv->forward_button, &child_allocation);
+ if (! priv->title_clickable) {
+ /* Keep a little distance from the navigation arrows. */
+ child_allocation.x += child_allocation.width + 5;
+
+ gtk_widget_size_request (priv->title_icon, &child_requisition);
+ child_allocation.width = child_requisition.width;
+ gtk_widget_size_allocate (priv->title_icon, &child_allocation);
+ }
+
return child_allocation.x + child_allocation.width;
}
@@ -333,34 +305,6 @@ size_allocate_label (EShellFolderTitleBar *title_bar,
*available_width_inout -= child_allocation.width;
}
-static void
-add_icon_widget (EShellFolderTitleBar *folder_title_bar)
-{
- EShellFolderTitleBarPrivate *priv;
- GdkPixmap *pixmap;
- GdkBitmap *mask;
-
- priv = folder_title_bar->priv;
-
- g_assert (priv->icon != NULL);
-
- gdk_pixbuf_render_pixmap_and_mask (priv->icon, &pixmap, &mask, 128);
-
- if (priv->icon_widget != NULL)
- gtk_widget_destroy (priv->icon_widget);
-
- priv->icon_widget = gtk_pixmap_new (pixmap, mask);
-
- gdk_pixmap_unref (pixmap);
- gdk_pixmap_unref (mask);
-
- gtk_misc_set_alignment (GTK_MISC (priv->icon_widget), 1.0, .5);
- gtk_misc_set_padding (GTK_MISC (priv->icon_widget), 0, 0);
-
- gtk_box_pack_start (GTK_BOX (folder_title_bar), priv->icon_widget, FALSE, TRUE, 2);
- gtk_widget_show (priv->icon_widget);
-}
-
/* The back/forward navigation buttons. */
@@ -441,7 +385,7 @@ title_button_toggled_cb (GtkToggleButton *title_button,
/* GtkObject methods. */
static void
-destroy (GtkObject *object)
+impl_destroy (GtkObject *object)
{
EShellFolderTitleBar *folder_title_bar;
EShellFolderTitleBarPrivate *priv;
@@ -460,40 +404,8 @@ destroy (GtkObject *object)
/* GTkWidget methods. */
static void
-realize (GtkWidget *widget)
-{
- EShellFolderTitleBar *folder_title_bar;
- EShellFolderTitleBarPrivate *priv;
-
- (* GTK_WIDGET_CLASS (parent_class)->realize) (widget);
-
- folder_title_bar = E_SHELL_FOLDER_TITLE_BAR (widget);
- priv = folder_title_bar->priv;
-
- if (priv->icon != NULL)
- add_icon_widget (E_SHELL_FOLDER_TITLE_BAR (widget));
-}
-
-static void
-unrealize (GtkWidget *widget)
-{
- EShellFolderTitleBar *folder_title_bar;
- EShellFolderTitleBarPrivate *priv;
-
- (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget);
-
- folder_title_bar = E_SHELL_FOLDER_TITLE_BAR (widget);
- priv = folder_title_bar->priv;
-
- if (priv->icon_widget != NULL) {
- gtk_widget_destroy (priv->icon_widget);
- priv->icon_widget = NULL;
- }
-}
-
-static void
-size_allocate (GtkWidget *widget,
- GtkAllocation *allocation)
+impl_size_allocate (GtkWidget *widget,
+ GtkAllocation *allocation)
{
EShellFolderTitleBar *title_bar;
EShellFolderTitleBarPrivate *priv;
@@ -509,10 +421,9 @@ size_allocate (GtkWidget *widget,
border_width = GTK_CONTAINER (widget)->border_width;
available_width = allocation->width - 2 * border_width;
- offset = size_allocate_navigation_buttons (title_bar, allocation);
+ offset = size_allocate_navigation_buttons_and_title_icon (title_bar, allocation);
available_width -= offset;
- size_allocate_icon (title_bar, allocation, & available_width);
width_before_icon = available_width;
if (priv->title_clickable)
@@ -538,12 +449,10 @@ class_init (EShellFolderTitleBarClass *klass)
GtkWidgetClass *widget_class;
object_class = GTK_OBJECT_CLASS (klass);
- object_class->destroy = destroy;
+ object_class->destroy = impl_destroy;
widget_class = GTK_WIDGET_CLASS (klass);
- widget_class->realize = realize;
- widget_class->unrealize = unrealize;
- widget_class->size_allocate = size_allocate;
+ widget_class->size_allocate = impl_size_allocate;
parent_class = gtk_type_class (PARENT_TYPE);
@@ -580,13 +489,17 @@ init (EShellFolderTitleBar *shell_folder_title_bar)
priv = g_new (EShellFolderTitleBarPrivate, 1);
priv->icon = NULL;
- priv->icon_widget = NULL;
+
+ priv->title_icon = NULL;
priv->title_label = NULL;
- priv->folder_bar_label = NULL;
- priv->title_button_label = NULL;
+
priv->title_button = NULL;
+ priv->title_button_icon = NULL;
+ priv->title_button_label = NULL;
priv->title_button_arrow = NULL;
+ priv->folder_bar_label = NULL;
+
priv->back_button = NULL;
priv->forward_button = NULL;
@@ -608,6 +521,7 @@ e_shell_folder_title_bar_construct (EShellFolderTitleBar *folder_title_bar)
EShellFolderTitleBarPrivate *priv;
GtkWidget *title_button_hbox;
GtkWidget *widget;
+ GtkRequisition button_requisition;
g_return_if_fail (folder_title_bar != NULL);
g_return_if_fail (E_IS_SHELL_FOLDER_TITLE_BAR (folder_title_bar));
@@ -615,25 +529,33 @@ e_shell_folder_title_bar_construct (EShellFolderTitleBar *folder_title_bar)
priv = folder_title_bar->priv;
widget = GTK_WIDGET (folder_title_bar);
+ priv->title_icon = new_empty_pixmap_widget ();
+ gtk_misc_set_alignment (GTK_MISC (priv->title_icon), 1.0, .5);
+ gtk_misc_set_padding (GTK_MISC (priv->title_icon), 2, 0);
+ gtk_widget_show (priv->title_icon);
+
priv->title_label = e_clipped_label_new ("");
- gtk_misc_set_padding (GTK_MISC (priv->title_label), 5, 0);
+ gtk_misc_set_padding (GTK_MISC (priv->title_label), 0, 0);
gtk_misc_set_alignment (GTK_MISC (priv->title_label), 0.0, 0.5);
set_title_bar_label_style (priv->title_label);
- /* make_bold (priv->title_label); */
priv->title_button_label = e_clipped_label_new ("");
gtk_misc_set_padding (GTK_MISC (priv->title_button_label), 2, 0);
gtk_misc_set_alignment (GTK_MISC (priv->title_button_label), 0.0, 0.5);
gtk_widget_show (priv->title_button_label);
set_title_bar_label_style (priv->title_button_label);
- /* make_bold (priv->title_label); */
priv->folder_bar_label = e_clipped_label_new ("");
gtk_misc_set_alignment (GTK_MISC (priv->folder_bar_label), 1.0, 0.5);
gtk_widget_show (priv->folder_bar_label);
set_title_bar_label_style (priv->folder_bar_label);
+ priv->title_button_icon = new_empty_pixmap_widget ();
+ gtk_widget_show (priv->title_button_icon);
+
title_button_hbox = gtk_hbox_new (FALSE, 0);
+ gtk_box_pack_start (GTK_BOX (title_button_hbox), priv->title_button_icon,
+ FALSE, TRUE, 2);
gtk_box_pack_start (GTK_BOX (title_button_hbox), priv->title_button_label,
TRUE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (title_button_hbox), create_pixmap_widget_from_xpm (down_arrow_xpm),
@@ -647,6 +569,7 @@ e_shell_folder_title_bar_construct (EShellFolderTitleBar *folder_title_bar)
gtk_widget_show (priv->title_button);
gtk_container_set_border_width (GTK_CONTAINER (folder_title_bar), 2);
+ gtk_box_pack_start (GTK_BOX (folder_title_bar), priv->title_icon, FALSE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (folder_title_bar), priv->title_label, FALSE, FALSE, 0);
gtk_box_pack_start (GTK_BOX (folder_title_bar), priv->title_button, FALSE, FALSE, 0);
gtk_box_pack_start (GTK_BOX (folder_title_bar), priv->folder_bar_label, TRUE, TRUE, 0);
@@ -656,8 +579,8 @@ e_shell_folder_title_bar_construct (EShellFolderTitleBar *folder_title_bar)
as the padding is hardcoded in GtkButton too (see CHILD_SPACING in
gtkbutton.c). */
gtk_misc_set_padding (GTK_MISC (priv->title_label),
- GTK_WIDGET (priv->title_button)->style->klass->xthickness + 3,
- GTK_WIDGET (priv->title_button)->style->klass->ythickness + 1);
+ GTK_WIDGET (priv->title_button)->style->klass->xthickness,
+ GTK_WIDGET (priv->title_button)->style->klass->ythickness + 2);
gtk_signal_connect (GTK_OBJECT (priv->title_button), "toggled",
GTK_SIGNAL_FUNC (title_button_toggled_cb), folder_title_bar);
@@ -760,9 +683,11 @@ e_shell_folder_title_bar_set_folder_bar_label (EShellFolderTitleBar *folder_titl
**/
void
e_shell_folder_title_bar_set_icon (EShellFolderTitleBar *folder_title_bar,
- const GdkPixbuf *icon)
+ GdkPixbuf *icon)
{
EShellFolderTitleBarPrivate *priv;
+ GdkPixmap *pixmap;
+ GdkBitmap *mask;
g_return_if_fail (icon != NULL);
@@ -771,13 +696,22 @@ e_shell_folder_title_bar_set_icon (EShellFolderTitleBar *folder_title_bar,
priv = folder_title_bar->priv;
- gdk_pixbuf_ref ((GdkPixbuf *) icon);
- if (priv->icon != NULL)
- gdk_pixbuf_unref (priv->icon);
- priv->icon = (GdkPixbuf *) icon;
+ if (icon == NULL) {
+ if (priv->icon != NULL)
+ gdk_pixbuf_unref (priv->icon);
+ priv->icon = new_empty_pixbuf ();
+ } else {
+ gdk_pixbuf_ref (icon);
+ if (priv->icon != NULL)
+ gdk_pixbuf_unref (priv->icon);
+ priv->icon = icon;
+ }
- if (priv->icon != NULL)
- add_icon_widget (folder_title_bar);
+ gdk_pixbuf_render_pixmap_and_mask (priv->icon, &pixmap, &mask, 127);
+ gtk_pixmap_set (GTK_PIXMAP (priv->title_button_icon), pixmap, mask);
+
+ gdk_pixbuf_render_pixmap_and_mask (priv->icon, &pixmap, &mask, 127);
+ gtk_pixmap_set (GTK_PIXMAP (priv->title_icon), pixmap, mask);
}
@@ -827,9 +761,13 @@ e_shell_folder_title_bar_set_title_clickable (EShellFolderTitleBar *folder_title
if (title_clickable) {
gtk_widget_hide (priv->title_label);
- gtk_widget_show (priv->title_button);
+ gtk_widget_hide (priv->title_icon);
+
+ gtk_widget_show_all (priv->title_button);
} else {
gtk_widget_hide (priv->title_button);
+
+ gtk_widget_show (priv->title_icon);
gtk_widget_show (priv->title_label);
}