/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* e-title-bar.c * * Copyright (C) 2000 Ximian, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. * * Author: Ettore Perazzoli */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <gtk/gtkbox.h> #include <gtk/gtkbutton.h> #include <gtk/gtkhbox.h> #include <gtk/gtklabel.h> #include <gtk/gtkpixmap.h> #include <gtk/gtksignal.h> #include <gtk/gtktypeutils.h> #include <gal/util/e-util.h> #include "e-clipped-label.h" #include "e-title-bar.h" enum { LABEL_BUTTON_PRESS_EVENT, BUTTON_CLICKED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; /* The pixmaps. */ static char *close_xpm[] = { "16 16 2 1", " c none", ". c #000000000000", " ", " ", " ", " .. .. ", " ... ... ", " ... ... ", " ...... ", " .... ", " .... ", " ...... ", " ... ... ", " ... ... ", " .. .. ", " ", " ", " " }; static char *pin_xpm[] = { "16 16 33 1", " c None", ". c #000000", "+ c #74744D", "@ c #B2B279", "# c #6C6C46", "$ c #CACA9C", "% c #F4F4AD", "& c #85855A", "* c #B1B175", "= c #9A9A66", "- c #A9A98A", "; c #B0B07B", "> c #535353", ", c #818181", "' c #B7B7B7", ") c #D8D8D8", "! c #FFFFFF", "~ c #EBEBA1", "{ c #8A8A75", "] c #9F9F76", "^ c #9E9E75", "/ c #8A8A66", "( c #979770", "_ c #6B6B46", ": c #28281A", "< c #505034", "[ c #666645", "} c #61614D", "| c #818155", "1 c #4A4A31", "2 c #4D4D34", "3 c #6C6C48", "4 c #5D5D3E", " ", " ", " ", " . . ", " .. .+. ", " .@...#$. ", " ......%&*=-;. ", ".>,')!.~{]^/(. ", " ......_:<[}|. ", " .1...23. ", " .. .4. ", " . . ", " ", " ", " ", " " }; #define PARENT_TYPE GTK_TYPE_FRAME static GtkFrameClass *parent_class = NULL; struct _ETitleBarPrivate { ETitleBarButtonMode button_mode; GtkWidget *label; GtkWidget *button; GtkWidget *close_gtk_pixmap; GtkWidget *pin_gtk_pixmap; }; /* Mode handling. We put both the close and pin GtkPixmaps into an hbox in the button, and hide one of them according to the mode. */ static void show_and_hide_pixmaps_according_to_mode (ETitleBar *title_bar) { ETitleBarPrivate *priv; priv = title_bar->priv; if (priv->close_gtk_pixmap == NULL || priv->pin_gtk_pixmap == NULL) return; switch (priv->button_mode) { case E_TITLE_BAR_BUTTON_MODE_PIN: gtk_widget_hide (priv->close_gtk_pixmap); gtk_widget_show (priv->pin_gtk_pixmap); break; case E_TITLE_BAR_BUTTON_MODE_CLOSE: gtk_widget_hide (priv->pin_gtk_pixmap); gtk_widget_show (priv->close_gtk_pixmap); break; default: g_assert_not_reached (); } } /* Child signal callbacks. */ static void button_realize_cb (GtkWidget *widget, gpointer data) { GdkPixmap *close_pixmap; GdkBitmap *close_mask; GdkPixmap *pin_pixmap; GdkBitmap *pin_mask; GtkWidget *hbox; ETitleBar *title_bar; ETitleBarPrivate *priv; title_bar = E_TITLE_BAR (data); priv = title_bar->priv; if (priv->close_gtk_pixmap != NULL) return; close_pixmap = gdk_pixmap_create_from_xpm_d (GTK_WIDGET (priv->button)->window, &close_mask, NULL, close_xpm); priv->close_gtk_pixmap = gtk_pixmap_new (close_pixmap, close_mask); pin_pixmap = gdk_pixmap_create_from_xpm_d (GTK_WIDGET (priv->button)->window, &pin_mask, NULL, pin_xpm); priv->pin_gtk_pixmap = gtk_pixmap_new (pin_pixmap, pin_mask); hbox = gtk_hbox_new (TRUE, 0); gtk_widget_show (hbox); gtk_box_pack_start (GTK_BOX (hbox), priv->pin_gtk_pixmap, TRUE, TRUE, 0); gtk_box_pack_start (GTK_BOX (hbox), priv->close_gtk_pixmap, TRUE, TRUE, 0); gtk_container_add (GTK_CONTAINER (priv->button), hbox); gdk_pixmap_unref (close_pixmap); g_object_unref (close_mask); gdk_pixmap_unref (pin_pixmap); g_object_unref (pin_mask); show_and_hide_pixmaps_according_to_mode (title_bar); } static void button_clicked_cb (GtkButton *button, gpointer data) { ETitleBar *title_bar; title_bar = E_TITLE_BAR (data); gtk_signal_emit (GTK_OBJECT (title_bar), signals[BUTTON_CLICKED]); } static void label_button_press_event_cb (GtkWidget *widget, GdkEventButton *event, gpointer data) { ETitleBar *title_bar; title_bar = E_TITLE_BAR (data); gtk_signal_emit (GTK_OBJECT (title_bar), signals[LABEL_BUTTON_PRESS_EVENT], event); } /* GObject methods. */ static void impl_finalize (GObject *object) { ETitleBar *title_bar; ETitleBarPrivate *priv; title_bar = E_TITLE_BAR (object); priv = title_bar->priv; g_free (priv); (* G_OBJECT_CLASS (parent_class)->finalize) (object); } static void class_init (ETitleBarClass *klass) { GObjectClass *object_class; object_class = G_OBJECT_CLASS (klass); object_class->finalize = impl_finalize; parent_class = g_type_class_ref(gtk_frame_get_type ()); signals[LABEL_BUTTON_PRESS_EVENT] = gtk_signal_new ("label_button_press_event", GTK_RUN_FIRST, GTK_CLASS_TYPE (object_class), G_STRUCT_OFFSET (ETitleBarClass, label_button_press_event), gtk_marshal_NONE__POINTER, GTK_TYPE_NONE, 1, GTK_TYPE_POINTER); signals[BUTTON_CLICKED] = gtk_signal_new ("button_clicked", GTK_RUN_FIRST, GTK_CLASS_TYPE (object_class), G_STRUCT_OFFSET (ETitleBarClass, button_clicked), gtk_marshal_NONE__NONE, GTK_TYPE_NONE, 0); } static void init (ETitleBar *title_bar) { ETitleBarPrivate *priv; priv = g_new (ETitleBarPrivate, 1); priv->button_mode = E_TITLE_BAR_BUTTON_MODE_CLOSE; priv->label = NULL; priv->button = NULL; priv->close_gtk_pixmap = NULL; priv->pin_gtk_pixmap = NULL; title_bar->priv = priv; } void e_title_bar_construct (ETitleBar *title_bar, const char *title) { ETitleBarPrivate *priv; GtkWidget *hbox; g_return_if_fail (title_bar != NULL); g_return_if_fail (E_IS_TITLE_BAR (title_bar)); priv = title_bar->priv; priv->label = e_clipped_label_new (title, PANGO_WEIGHT_BOLD, 1.0); gtk_misc_set_alignment (GTK_MISC (priv->label), 0.0, 0.5); gtk_misc_set_padding (GTK_MISC (priv->label), 2, 0); gtk_widget_show (priv->label); priv->button = gtk_button_new (); GTK_WIDGET_UNSET_FLAGS (priv->button, GTK_CAN_FOCUS); gtk_container_set_border_width (GTK_CONTAINER (priv->button), 1); gtk_button_set_relief (GTK_BUTTON (priv->button), GTK_RELIEF_NONE); gtk_widget_show (priv->button); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), priv->label, TRUE, TRUE, 2); gtk_box_pack_start (GTK_BOX (hbox), priv->button, FALSE, TRUE, 1); gtk_widget_show (hbox); gtk_container_add (GTK_CONTAINER (title_bar), hbox); g_signal_connect((priv->button), "realize", G_CALLBACK (button_realize_cb), title_bar); g_signal_connect((priv->button), "clicked", G_CALLBACK (button_clicked_cb), title_bar); g_signal_connect((priv->label), "button_press_event", G_CALLBACK (label_button_press_event_cb), title_bar); } GtkWidget * e_title_bar_new (const char *title) { ETitleBar *title_bar; title_bar = gtk_type_new (e_title_bar_get_type ()); e_title_bar_construct (title_bar, title); return GTK_WIDGET (title_bar); } void e_title_bar_set_title (ETitleBar *title_bar, const char *title) { g_return_if_fail (title_bar != NULL); g_return_if_fail (E_IS_TITLE_BAR (title_bar)); e_clipped_label_set_text (E_CLIPPED_LABEL (title_bar->priv->label), title); } void e_title_bar_show_button (ETitleBar *title_bar, gboolean show) { ETitleBarPrivate *priv; g_return_if_fail (title_bar != NULL); g_return_if_fail (E_IS_TITLE_BAR (title_bar)); priv = title_bar->priv; if (show) gtk_widget_show (priv->button); else gtk_widget_hide (priv->button); } void e_title_bar_set_button_mode (ETitleBar *title_bar, ETitleBarButtonMode button_mode) { ETitleBarPrivate *priv; g_return_if_fail (title_bar != NULL); g_return_if_fail (E_IS_TITLE_BAR (title_bar)); g_return_if_fail (button_mode == E_TITLE_BAR_BUTTON_MODE_CLOSE || button_mode == E_TITLE_BAR_BUTTON_MODE_PIN); priv = title_bar->priv; if (priv->button_mode == button_mode) return; priv->button_mode = button_mode; show_and_hide_pixmaps_according_to_mode (title_bar); } ETitleBarButtonMode e_title_bar_get_button_mode (ETitleBar *title_bar) { g_return_val_if_fail (title_bar != NULL, E_TITLE_BAR_BUTTON_MODE_CLOSE); g_return_val_if_fail (E_IS_TITLE_BAR (title_bar), E_TITLE_BAR_BUTTON_MODE_CLOSE); return title_bar->priv->button_mode; } E_MAKE_TYPE (e_title_bar, "ETitleBar", ETitleBar, class_init, init, PARENT_TYPE)