/* GTK - The GIMP Toolkit * Copyright (C) 2001 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ /* * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS * file for a list of people on the GTK+ Team. See the ChangeLog * files for a list of changes. These files are distributed with * GTK+ at ftp://ftp.gtk.org/pub/gtk/. */ #include #include "gtkalias.h" #include "gtkimagemenuitem.h" #include "gtkaccellabel.h" #include "gtkintl.h" #include "gtkstock.h" #include "gtkiconfactory.h" #include "gtkimage.h" #include "gtkcontainer.h" #include "gtkwindow.h" static void gtk_image_menu_item_class_init (GtkImageMenuItemClass *klass); static void gtk_image_menu_item_init (GtkImageMenuItem *image_menu_item); static void gtk_image_menu_item_size_request (GtkWidget *widget, GtkRequisition *requisition); static void gtk_image_menu_item_size_allocate (GtkWidget *widget, GtkAllocation *allocation); static void gtk_image_menu_item_remove (GtkContainer *container, GtkWidget *child); static void gtk_image_menu_item_toggle_size_request (GtkMenuItem *menu_item, gint *requisition); static void gtk_image_menu_item_forall (GtkContainer *container, gboolean include_internals, GtkCallback callback, gpointer callback_data); static void gtk_image_menu_item_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void gtk_image_menu_item_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void gtk_image_menu_item_screen_changed (GtkWidget *widget, GdkScreen *previous_screen); enum { PROP_ZERO, PROP_IMAGE }; static GtkMenuItemClass *parent_class = NULL; GType gtk_image_menu_item_get_type (void) { static GType image_menu_item_type = 0; if (!image_menu_item_type) { static const GTypeInfo image_menu_item_info = { sizeof (GtkImageMenuItemClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) gtk_image_menu_item_class_init, NULL, /* class_finalize */ NULL, /* class_data */ sizeof (GtkImageMenuItem), 0, /* n_preallocs */ (GInstanceInitFunc) gtk_image_menu_item_init, }; image_menu_item_type = g_type_register_static (GTK_TYPE_MENU_ITEM, "GtkImageMenuItem", &image_menu_item_info, 0); } return image_menu_item_type; } static void gtk_image_menu_item_class_init (GtkImageMenuItemClass *klass) { GObjectClass *gobject_class; GtkWidgetClass *widget_class; GtkMenuItemClass *menu_item_class; GtkContainerClass *container_class; gobject_class = (GObjectClass*) klass; widget_class = (GtkWidgetClass*) klass; menu_item_class = (GtkMenuItemClass*) klass; container_class = (GtkContainerClass*) klass; parent_class = g_type_class_peek_parent (klass); widget_class->screen_changed = gtk_image_menu_item_screen_changed; widget_class->size_request = gtk_image_menu_item_size_request; widget_class->size_allocate = gtk_image_menu_item_size_allocate; container_class->forall = gtk_image_menu_item_forall; container_class->remove = gtk_image_menu_item_remove; menu_item_class->toggle_size_request = gtk_image_menu_item_toggle_size_request; gobject_class->set_property = gtk_image_menu_item_set_property; gobject_class->get_property = gtk_image_menu_item_get_property; g_object_class_install_property (gobject_class, PROP_IMAGE, g_param_spec_object ("image", P_("Image widget"), P_("Child widget to appear next to the menu text"), GTK_TYPE_WIDGET, G_PARAM_READABLE | G_PARAM_WRITABLE)); gtk_settings_install_property (g_param_spec_boolean ("gtk-menu-images", P_("Show menu images"), P_("Whether images should be shown in menus"), TRUE, G_PARAM_READWRITE)); } static void gtk_image_menu_item_init (GtkImageMenuItem *image_menu_item) { image_menu_item->image = NULL; } static void gtk_image_menu_item_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtkImageMenuItem *image_menu_item = GTK_IMAGE_MENU_ITEM (object); switch (prop_id) { case PROP_IMAGE: { GtkWidget *image; image = (GtkWidget*) g_value_get_object (value); gtk_image_menu_item_set_image (image_menu_item, image); } break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gtk_image_menu_item_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkImageMenuItem *image_menu_item = GTK_IMAGE_MENU_ITEM (object); switch (prop_id) { case PROP_IMAGE: g_value_set_object (value, (GObject*) image_menu_item->image); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean show_image (GtkImageMenuItem *image_menu_item) { GtkSettings *settings = gtk_widget_get_settings (GTK_WIDGET (image_menu_item)); gboolean show; g_object_get (settings, "gtk-menu-images", &show, NULL); return show; } static void gtk_image_menu_item_toggle_size_request (GtkMenuItem *menu_item, gint *requisition) { GtkImageMenuItem *image_menu_item = GTK_IMAGE_MENU_ITEM (menu_item); *requisition = 0; if (image_menu_item->image && show_image (image_menu_item)) { GtkRequisition image_requisition; gtk_widget_get_child_requisition (image_menu_item->image, &image_requisition); if (image_requisition.width > 0) { guint toggle_spacing; gtk_widget_style_get (GTK_WIDGET (menu_item), "toggle_spacing", &toggle_spacing, NULL); *requisition = image_requisition.width + toggle_spacing; } } } static void gtk_image_menu_item_size_request (GtkWidget *widget, GtkRequisition *requisition) { GtkImageMenuItem *image_menu_item; gint child_height = 0; image_menu_item = GTK_IMAGE_MENU_ITEM (widget); if (image_menu_item->image && GTK_WIDGET_VISIBLE (image_menu_item->image) && show_image (image_menu_item)) { GtkRequisition child_requisition; gtk_widget_size_request (image_menu_item->image, &child_requisition); child_height = child_requisition.height; } (* GTK_WIDGET_CLASS (parent_class)->size_request) (widget, requisition); /* not done with height since that happens via the * toggle_size_request */ requisition->height = MAX (requisition->height, child_height); /* Note that GtkMenuShell always size requests before * toggle_size_request, so toggle_size_request will be able to use * image_menu_item->image->requisition */ } static void gtk_image_menu_item_size_allocate (GtkWidget *widget, GtkAllocation *allocation) { GtkImageMenuItem *image_menu_item; image_menu_item = GTK_IMAGE_MENU_ITEM (widget); (* GTK_WIDGET_CLASS (parent_class)->size_allocate) (widget, allocation); if (image_menu_item->image && show_image (image_menu_item)) { gint x, y, offset; GtkRequisition child_requisition; GtkAllocation child_allocation; guint horizontal_padding, toggle_spacing; gtk_widget_style_get (widget, "horizontal_padding", &horizontal_padding, "toggle_spacing", &toggle_spacing, NULL); /* Man this is lame hardcoding action, but I can't * come up with a solution that's really better. */ gtk_widget_get_child_requisition (image_menu_item->image, &child_requisition); offset = GTK_CONTAINER (image_menu_item)->border_width + widget->style->xthickness; if (gtk_widget_get_direction (widget) == GTK_TEXT_DIR_LTR) { x = offset + horizontal_padding + (GTK_MENU_ITEM (image_menu_item)->toggle_size - toggle_spacing - child_requisition.width) / 2; } else { x = widget->allocation.width - offset - horizontal_padding - GTK_MENU_ITEM (image_menu_item)->toggle_size + toggle_spacing + (GTK_MENU_ITEM (image_menu_item)->toggle_size - toggle_spacing - child_requisition.width) / 2; } y = (widget->allocation.height - child_requisition.height) / 2; child_allocation.width = child_requisition.width; child_allocation.height = child_requisition.height; child_allocation.x = widget->allocation.x + MAX (x, 0); child_allocation.y = widget->allocation.y + MAX (y, 0); gtk_widget_size_allocate (image_menu_item->image, &child_allocation); } } static void gtk_image_menu_item_forall (GtkContainer *container, gboolean include_internals, GtkCallback callback, gpointer callback_data) { GtkImageMenuItem *image_menu_item = GTK_IMAGE_MENU_ITEM (container); (* GTK_CONTAINER_CLASS (parent_class)->forall) (container, include_internals, callback, callback_data); if (image_menu_item->image) (* callback) (image_menu_item->image, callback_data); } /** * gtk_image_menu_item_new: * @returns: a new #GtkImageMenuItem. * * Creates a new #GtkImageMenuItem with an empty label. **/ GtkWidget* gtk_image_menu_item_new (void) { return g_object_new (GTK_TYPE_IMAGE_MENU_ITEM, NULL); } /** * gtk_image_menu_item_new_with_label: * @label: the text of the menu item. * @returns: a new #GtkImageMenuItem. * * Creates a new #GtkImageMenuItem containing a label. **/ GtkWidget* gtk_image_menu_item_new_with_label (const gchar *label) { GtkImageMenuItem *image_menu_item; GtkWidget *accel_label; image_menu_item = g_object_new (GTK_TYPE_IMAGE_MENU_ITEM, NULL); accel_label = gtk_accel_label_new (label); gtk_misc_set_alignment (GTK_MISC (accel_label), 0.0, 0.5); gtk_container_add (GTK_CONTAINER (image_menu_item), accel_label); gtk_accel_label_set_accel_widget (GTK_ACCEL_LABEL (accel_label), GTK_WIDGET (image_menu_item)); gtk_widget_show (accel_label); return GTK_WIDGET(image_menu_item); } /** * gtk_image_menu_item_new_with_mnemonic: * @label: the text of the menu item, with an underscore in front of the * mnemonic character * @returns: a new #GtkImageMenuItem * * Creates a new #GtkImageMenuItem containing a label. The label * will be created using gtk_label_new_with_mnemonic(), so underscores * in @label indicate the mnemonic for the menu item. **/ GtkWidget* gtk_image_menu_item_new_with_mnemonic (const gchar *label) { GtkImageMenuItem *image_menu_item; GtkWidget *accel_label; image_menu_item = g_object_new (GTK_TYPE_IMAGE_MENU_ITEM, NULL); accel_label = g_object_new (GTK_TYPE_ACCEL_LABEL, NULL); gtk_label_set_text_with_mnemonic (GTK_LABEL (accel_label), label); gtk_misc_set_alignment (GTK_MISC (accel_label), 0.0, 0.5); gtk_container_add (GTK_CONTAINER (image_menu_item), accel_label); gtk_accel_label_set_accel_widget (GTK_ACCEL_LABEL (accel_label), GTK_WIDGET (image_menu_item)); gtk_widget_show (accel_label); return GTK_WIDGET(image_menu_item); } /** * gtk_image_menu_item_new_from_stock: * @stock_id: the name of the stock item. * @accel_group: the #GtkAccelGroup to add the menu items accelerator to, * or %NULL. * @returns: a new #GtkImageMenuItem. * * Creates a new #GtkImageMenuItem containing the image and text from a * stock item. Some stock ids have preprocessor macros like #GTK_STOCK_OK * and #GTK_STOCK_APPLY. * * If you want this menu item to have changeable accelerators, then pass in * %NULL for accel_group. Next call gtk_menu_item_set_accel_path() with an * appropriate path for the menu item, use gtk_stock_lookup() to look up the * standard accelerator for the stock item, and if one is found, call * gtk_accel_map_add_entry() to register it. **/ GtkWidget* gtk_image_menu_item_new_from_stock (const gchar *stock_id, GtkAccelGroup *accel_group) { GtkWidget *image; GtkStockItem stock_item; GtkWidget *item; g_return_val_if_fail (stock_id != NULL, NULL); image = gtk_image_new_from_stock (stock_id, GTK_ICON_SIZE_MENU); if (gtk_stock_lookup (stock_id, &stock_item)) { item = gtk_image_menu_item_new_with_mnemonic (stock_item.label); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image); if (stock_item.keyval && accel_group) gtk_widget_add_accelerator (item, "activate", accel_group, stock_item.keyval, stock_item.modifier, GTK_ACCEL_VISIBLE); } else { item = gtk_image_menu_item_new_with_mnemonic (stock_id); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image); } gtk_widget_show (image); return item; } /** * gtk_image_menu_item_set_image: * @image_menu_item: a #GtkImageMenuItem. * @image: a widget to set as the image for the menu item. * * Sets the image of @image_menu_item to the given widget. * Note that it depends on the show-menu-images setting whether * the image will be displayed or not. **/ void gtk_image_menu_item_set_image (GtkImageMenuItem *image_menu_item, GtkWidget *image) { g_return_if_fail (GTK_IS_IMAGE_MENU_ITEM (image_menu_item)); if (image == image_menu_item->image) return; if (image_menu_item->image) gtk_container_remove (GTK_CONTAINER (image_menu_item), image_menu_item->image); image_menu_item->image = image; if (image == NULL) return; gtk_widget_set_parent (image, GTK_WIDGET (image_menu_item)); g_object_set (image, "visible", show_image (image_menu_item), "no_show_all", TRUE, NULL); g_object_notify (G_OBJECT (image_menu_item), "image"); } /** * gtk_image_menu_item_get_image: * @image_menu_item: a #GtkImageMenuItem. * @returns: the widget set as image of @image_menu_item. * * Gets the widget that is currently set as the image of @image_menu_item. * See gtk_image_menu_item_set_image(). **/ GtkWidget* gtk_image_menu_item_get_image (GtkImageMenuItem *image_menu_item) { g_return_val_if_fail (GTK_IS_IMAGE_MENU_ITEM (image_menu_item), NULL); return image_menu_item->image; } static void gtk_image_menu_item_remove (GtkContainer *container, GtkWidget *child) { GtkImageMenuItem *image_menu_item; image_menu_item = GTK_IMAGE_MENU_ITEM (container); if (child == image_menu_item->image) { gboolean widget_was_visible; widget_was_visible = GTK_WIDGET_VISIBLE (child); gtk_widget_unparent (child); image_menu_item->image = NULL; if (GTK_WIDGET_VISIBLE (container) && widget_was_visible) gtk_widget_queue_resize (GTK_WIDGET (container)); g_object_notify (G_OBJECT (image_menu_item), "image"); } else { (* GTK_CONTAINER_CLASS (parent_class)->remove) (container, child); } } static void show_image_change_notify (GtkImageMenuItem *image_menu_item) { if (image_menu_item->image) { if (show_image (image_menu_item)) gtk_widget_show (image_menu_item->image); else gtk_widget_hide (image_menu_item->image); } } static void traverse_container (GtkWidget *widget, gpointer data) { if (GTK_IS_IMAGE_MENU_ITEM (widget)) show_image_change_notify (GTK_IMAGE_MENU_ITEM (widget)); else if (GTK_IS_CONTAINER (widget)) gtk_container_forall (GTK_CONTAINER (widget), traverse_container, NULL); } static void gtk_image_menu_item_setting_changed (GtkSettings *settings) { GList *list; list = gtk_window_list_toplevels (); for (; list; list = list->next) gtk_container_forall (GTK_CONTAINER (list->data), traverse_container, NULL); g_list_free (list); } static void gtk_image_menu_item_screen_changed (GtkWidget *widget, GdkScreen *previous_screen) { GtkSettings *settings; guint show_image_connection; if (!gtk_widget_has_screen (widget)) return; settings = gtk_widget_get_settings (widget); show_image_connection = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (settings), "gtk-image-menu-item-connection")); if (show_image_connection) return; show_image_connection = g_signal_connect (settings, "notify::gtk-menu-images", G_CALLBACK (gtk_image_menu_item_setting_changed), 0); g_object_set_data (G_OBJECT (settings), "gtk-image-menu-item-connection", GINT_TO_POINTER (show_image_connection)); show_image_change_notify (GTK_IMAGE_MENU_ITEM (widget)); }