diff options
Diffstat (limited to 'gtk/gtkmenu.c')
-rw-r--r-- | gtk/gtkmenu.c | 732 |
1 files changed, 732 insertions, 0 deletions
diff --git a/gtk/gtkmenu.c b/gtk/gtkmenu.c new file mode 100644 index 000000000..13fff9023 --- /dev/null +++ b/gtk/gtkmenu.c @@ -0,0 +1,732 @@ +/* GTK - The GIMP Toolkit + * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the Free + * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + */ +#include <ctype.h> +#include "gdk/gdkkeysyms.h" +#include "gtkmain.h" +#include "gtkmenu.h" +#include "gtkmenuitem.h" +#include "gtksignal.h" + + +#define MENU_ITEM_CLASS(w) GTK_MENU_ITEM_CLASS (GTK_OBJECT (w)->klass) + + +static void gtk_menu_class_init (GtkMenuClass *klass); +static void gtk_menu_init (GtkMenu *menu); +static void gtk_menu_show (GtkWidget *widget); +static void gtk_menu_map (GtkWidget *widget); +static void gtk_menu_unmap (GtkWidget *widget); +static void gtk_menu_realize (GtkWidget *widget); +static void gtk_menu_size_request (GtkWidget *widget, + GtkRequisition *requisition); +static void gtk_menu_size_allocate (GtkWidget *widget, + GtkAllocation *allocation); +static void gtk_menu_paint (GtkWidget *widget); +static void gtk_menu_draw (GtkWidget *widget, + GdkRectangle *area); +static gint gtk_menu_expose (GtkWidget *widget, + GdkEventExpose *event); +static gint gtk_menu_configure (GtkWidget *widget, + GdkEventConfigure *event); +static gint gtk_menu_key_press (GtkWidget *widget, + GdkEventKey *event); +static gint gtk_menu_need_resize (GtkContainer *container); +static void gtk_menu_deactivate (GtkMenuShell *menu_shell); + + +guint +gtk_menu_get_type () +{ + static guint menu_type = 0; + + if (!menu_type) + { + GtkTypeInfo menu_info = + { + "GtkMenu", + sizeof (GtkMenu), + sizeof (GtkMenuClass), + (GtkClassInitFunc) gtk_menu_class_init, + (GtkObjectInitFunc) gtk_menu_init, + (GtkArgFunc) NULL, + }; + + menu_type = gtk_type_unique (gtk_menu_shell_get_type (), &menu_info); + } + + return menu_type; +} + +static void +gtk_menu_class_init (GtkMenuClass *class) +{ + GtkWidgetClass *widget_class; + GtkContainerClass *container_class; + GtkMenuShellClass *menu_shell_class; + + widget_class = (GtkWidgetClass*) class; + container_class = (GtkContainerClass*) class; + menu_shell_class = (GtkMenuShellClass*) class; + + widget_class->show = gtk_menu_show; + widget_class->map = gtk_menu_map; + widget_class->unmap = gtk_menu_unmap; + widget_class->realize = gtk_menu_realize; + widget_class->draw = gtk_menu_draw; + widget_class->size_request = gtk_menu_size_request; + widget_class->size_allocate = gtk_menu_size_allocate; + widget_class->expose_event = gtk_menu_expose; + widget_class->configure_event = gtk_menu_configure; + widget_class->key_press_event = gtk_menu_key_press; + + container_class->need_resize = gtk_menu_need_resize; + + menu_shell_class->submenu_placement = GTK_LEFT_RIGHT; + menu_shell_class->deactivate = gtk_menu_deactivate; +} + +static void +gtk_menu_init (GtkMenu *menu) +{ + GTK_WIDGET_SET_FLAGS (menu, GTK_ANCHORED); + + menu->parent_menu_item = NULL; + menu->old_active_menu_item = NULL; + menu->accelerator_table = NULL; + menu->position_func = NULL; + menu->position_func_data = NULL; + + GTK_MENU_SHELL (menu)->menu_flag = TRUE; +} + +GtkWidget* +gtk_menu_new () +{ + return GTK_WIDGET (gtk_type_new (gtk_menu_get_type ())); +} + +void +gtk_menu_append (GtkMenu *menu, + GtkWidget *child) +{ + gtk_menu_shell_append (GTK_MENU_SHELL (menu), child); +} + +void +gtk_menu_prepend (GtkMenu *menu, + GtkWidget *child) +{ + gtk_menu_shell_prepend (GTK_MENU_SHELL (menu), child); +} + +void +gtk_menu_insert (GtkMenu *menu, + GtkWidget *child, + gint position) +{ + gtk_menu_shell_insert (GTK_MENU_SHELL (menu), child, position); +} + +void +gtk_menu_popup (GtkMenu *menu, + GtkWidget *parent_menu_shell, + GtkWidget *parent_menu_item, + GtkMenuPositionFunc func, + gpointer data, + gint button, + guint32 activate_time) +{ + g_return_if_fail (menu != NULL); + g_return_if_fail (GTK_IS_MENU (menu)); + + GTK_MENU_SHELL (menu)->parent_menu_shell = parent_menu_shell; + GTK_MENU_SHELL (menu)->active = TRUE; + GTK_MENU_SHELL (menu)->button = button; + + menu->parent_menu_item = parent_menu_item; + menu->position_func = func; + menu->position_func_data = data; + GTK_MENU_SHELL (menu)->activate_time = activate_time; + + gtk_widget_show (GTK_WIDGET (menu)); + gtk_grab_add (GTK_WIDGET (menu)); +} + +void +gtk_menu_popdown (GtkMenu *menu) +{ + GtkMenuShell *menu_shell; + + g_return_if_fail (menu != NULL); + g_return_if_fail (GTK_IS_MENU (menu)); + + menu_shell = GTK_MENU_SHELL (menu); + + menu_shell->parent_menu_shell = NULL; + menu_shell->active = FALSE; + + if (menu_shell->active_menu_item) + { + menu->old_active_menu_item = menu_shell->active_menu_item; + gtk_menu_item_deselect (GTK_MENU_ITEM (menu_shell->active_menu_item)); + menu_shell->active_menu_item = NULL; + } + + gtk_widget_hide (GTK_WIDGET (menu)); + gtk_grab_remove (GTK_WIDGET (menu)); +} + +GtkWidget* +gtk_menu_get_active (GtkMenu *menu) +{ + GtkWidget *child; + GList *children; + + g_return_val_if_fail (menu != NULL, NULL); + g_return_val_if_fail (GTK_IS_MENU (menu), NULL); + + if (!menu->old_active_menu_item) + { + child = NULL; + children = GTK_MENU_SHELL (menu)->children; + + while (children) + { + child = children->data; + children = children->next; + + if (GTK_BIN (child)->child) + break; + child = NULL; + } + + menu->old_active_menu_item = child; + } + + return menu->old_active_menu_item; +} + +void +gtk_menu_set_active (GtkMenu *menu, + gint index) +{ + GtkWidget *child; + GList *tmp_list; + + g_return_if_fail (menu != NULL); + g_return_if_fail (GTK_IS_MENU (menu)); + + tmp_list = g_list_nth (GTK_MENU_SHELL (menu)->children, index); + if (tmp_list) + { + child = tmp_list->data; + if (GTK_BIN (child)->child) + menu->old_active_menu_item = child; + } +} + +void +gtk_menu_set_accelerator_table (GtkMenu *menu, + GtkAcceleratorTable *table) +{ + g_return_if_fail (menu != NULL); + g_return_if_fail (GTK_IS_MENU (menu)); + + if (menu->accelerator_table) + gtk_accelerator_table_unref (menu->accelerator_table); + + menu->accelerator_table = table; + if (menu->accelerator_table) + gtk_accelerator_table_ref (menu->accelerator_table); +} + + +static void +gtk_menu_show (GtkWidget *widget) +{ + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + + GTK_WIDGET_SET_FLAGS (widget, GTK_VISIBLE); + gtk_widget_map (widget); +} + +static void +gtk_menu_map (GtkWidget *widget) +{ + GtkMenu *menu; + GtkMenuShell *menu_shell; + GtkWidget *child; + GList *children; + GtkAllocation allocation; + gint x, y; + + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + + menu = GTK_MENU (widget); + menu_shell = GTK_MENU_SHELL (widget); + GTK_WIDGET_SET_FLAGS (menu_shell, GTK_MAPPED); + GTK_WIDGET_UNSET_FLAGS (menu_shell, GTK_UNMAPPED); + + gtk_widget_size_request (widget, &widget->requisition); + + if (menu_shell->menu_flag) + { + menu_shell->menu_flag = FALSE; + + allocation.x = widget->allocation.x; + allocation.y = widget->allocation.y; + allocation.width = widget->requisition.width; + allocation.height = widget->requisition.height; + + gtk_widget_size_allocate (widget, &allocation); + } + + gdk_window_get_pointer (NULL, &x, &y, NULL); + + if (menu->position_func) + (* menu->position_func) (menu, &x, &y, menu->position_func_data); + else + { + gint screen_width; + gint screen_height; + + screen_width = gdk_screen_width (); + screen_height = gdk_screen_height (); + + x -= 2; + y -= 2; + + if ((x + widget->requisition.width) > screen_width) + x -= ((x + widget->requisition.width) - screen_width); + if (x < 0) + x = 0; + if ((y + widget->requisition.height) > screen_height) + y -= ((y + widget->requisition.height) - screen_height); + if (y < 0) + y = 0; + } + + gdk_window_move_resize (widget->window, x, y, + widget->requisition.width, + widget->requisition.height); + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + if (GTK_WIDGET_VISIBLE (child) && !GTK_WIDGET_MAPPED (child)) + gtk_widget_map (child); + } + + gdk_window_show (widget->window); +} + +static void +gtk_menu_unmap (GtkWidget *widget) +{ + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + + GTK_WIDGET_UNSET_FLAGS (widget, GTK_MAPPED); + GTK_WIDGET_SET_FLAGS (widget, GTK_UNMAPPED); + gdk_window_hide (widget->window); +} + +static void +gtk_menu_realize (GtkWidget *widget) +{ + GdkWindowAttr attributes; + gint attributes_mask; + + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + + GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED); + + attributes.x = widget->allocation.x; + attributes.y = widget->allocation.y; + attributes.width = widget->allocation.width; + attributes.height = widget->allocation.height; + attributes.wclass = GDK_INPUT_OUTPUT; + attributes.visual = gtk_widget_get_visual (widget); + attributes.colormap = gtk_widget_get_colormap (widget); + attributes.window_type = GDK_WINDOW_TEMP; + attributes.event_mask = gtk_widget_get_events (widget); + attributes.event_mask |= (GDK_EXPOSURE_MASK | + GDK_KEY_PRESS_MASK | + GDK_STRUCTURE_MASK); + + attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP; + widget->window = gdk_window_new (NULL, &attributes, attributes_mask); + gdk_window_set_user_data (widget->window, widget); + + widget->style = gtk_style_attach (widget->style, widget->window); + gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL); +} + +static void +gtk_menu_size_request (GtkWidget *widget, + GtkRequisition *requisition) +{ + GtkMenu *menu; + GtkMenuShell *menu_shell; + GtkWidget *child; + GList *children; + gint max_accelerator_size; + gint max_toggle_size; + + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + g_return_if_fail (requisition != NULL); + + menu = GTK_MENU (widget); + menu_shell = GTK_MENU_SHELL (widget); + + requisition->width = 0; + requisition->height = 0; + + max_accelerator_size = 0; + max_toggle_size = 0; + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + if (GTK_WIDGET_VISIBLE (child)) + { + GTK_MENU_ITEM (child)->show_submenu_indicator = TRUE; + gtk_widget_size_request (child, &child->requisition); + + requisition->width = MAX (requisition->width, child->requisition.width); + requisition->height += child->requisition.height; + + max_accelerator_size = MAX (max_accelerator_size, GTK_MENU_ITEM (child)->accelerator_size); + max_toggle_size = MAX (max_toggle_size, MENU_ITEM_CLASS (child)->toggle_size); + } + } + + requisition->width += max_toggle_size + max_accelerator_size; + requisition->width += (GTK_CONTAINER (menu)->border_width + + widget->style->klass->xthickness) * 2; + requisition->height += (GTK_CONTAINER (menu)->border_width + + widget->style->klass->ythickness) * 2; + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + GTK_MENU_ITEM (child)->accelerator_size = max_accelerator_size; + GTK_MENU_ITEM (child)->toggle_size = max_toggle_size; + } +} + +static void +gtk_menu_size_allocate (GtkWidget *widget, + GtkAllocation *allocation) +{ + GtkMenu *menu; + GtkMenuShell *menu_shell; + GtkWidget *child; + GtkAllocation child_allocation; + GList *children; + + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + g_return_if_fail (allocation != NULL); + + menu = GTK_MENU (widget); + menu_shell = GTK_MENU_SHELL (widget); + + widget->allocation = *allocation; + + if (menu_shell->children) + { + child_allocation.x = (GTK_CONTAINER (menu)->border_width + + widget->style->klass->xthickness); + child_allocation.y = (GTK_CONTAINER (menu)->border_width + + widget->style->klass->ythickness); + child_allocation.width = allocation->width - child_allocation.x * 2; + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + if (GTK_WIDGET_VISIBLE (child)) + { + child_allocation.height = child->requisition.height; + + gtk_widget_size_allocate (child, &child_allocation); + + child_allocation.y += child_allocation.height; + } + } + } +} + +static void +gtk_menu_paint (GtkWidget *widget) +{ + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + + if (GTK_WIDGET_DRAWABLE (widget)) + { + gtk_draw_shadow (widget->style, + widget->window, + GTK_STATE_NORMAL, + GTK_SHADOW_OUT, + 0, 0, + widget->allocation.width, + widget->allocation.height); + } +} + +static void +gtk_menu_draw (GtkWidget *widget, + GdkRectangle *area) +{ + GtkMenuShell *menu_shell; + GtkWidget *child; + GdkRectangle child_area; + GList *children; + + g_return_if_fail (widget != NULL); + g_return_if_fail (GTK_IS_MENU (widget)); + g_return_if_fail (area != NULL); + + if (GTK_WIDGET_DRAWABLE (widget)) + { + gtk_menu_paint (widget); + + menu_shell = GTK_MENU_SHELL (widget); + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + if (gtk_widget_intersect (child, area, &child_area)) + gtk_widget_draw (child, &child_area); + } + } +} + +static gint +gtk_menu_expose (GtkWidget *widget, + GdkEventExpose *event) +{ + GtkMenuShell *menu_shell; + GtkWidget *child; + GdkEventExpose child_event; + GList *children; + + g_return_val_if_fail (widget != NULL, FALSE); + g_return_val_if_fail (GTK_IS_MENU (widget), FALSE); + g_return_val_if_fail (event != NULL, FALSE); + + if (!GTK_WIDGET_UNMAPPED (widget)) + GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED); + + if (GTK_WIDGET_DRAWABLE (widget)) + { + gtk_menu_paint (widget); + + menu_shell = GTK_MENU_SHELL (widget); + child_event = *event; + + children = menu_shell->children; + while (children) + { + child = children->data; + children = children->next; + + if (GTK_WIDGET_NO_WINDOW (child) && + gtk_widget_intersect (child, &event->area, &child_event.area)) + gtk_widget_event (child, (GdkEvent*) &child_event); + } + } + + return FALSE; +} + +static gint +gtk_menu_configure (GtkWidget *widget, + GdkEventConfigure *event) +{ + GtkAllocation allocation; + + g_return_val_if_fail (widget != NULL, FALSE); + g_return_val_if_fail (GTK_IS_MENU (widget), FALSE); + g_return_val_if_fail (event != NULL, FALSE); + + if (GTK_MENU_SHELL (widget)->menu_flag) + { + GTK_MENU_SHELL (widget)->menu_flag = FALSE; + + allocation.x = 0; + allocation.y = 0; + allocation.width = event->width; + allocation.height = event->height; + + gtk_widget_size_allocate (widget, &allocation); + } + + return FALSE; +} + +static gint +gtk_menu_key_press (GtkWidget *widget, + GdkEventKey *event) +{ + GtkAllocation allocation; + GtkAcceleratorTable *table; + GtkMenuShell *menu_shell; + GtkMenuItem *menu_item; + gchar *signame; + int delete; + + g_return_val_if_fail (widget != NULL, FALSE); + g_return_val_if_fail (GTK_IS_MENU (widget), FALSE); + g_return_val_if_fail (event != NULL, FALSE); + + delete = ((event->keyval == GDK_Delete) || + (event->keyval == GDK_BackSpace)); + + if (delete || ((event->keyval >= 0x20) && (event->keyval <= 0xFF))) + { + menu_shell = GTK_MENU_SHELL (widget); + menu_item = GTK_MENU_ITEM (menu_shell->active_menu_item); + + if (menu_item && GTK_BIN (menu_item)->child) + { + /* block resizes */ + gtk_container_block_resize (GTK_CONTAINER (widget)); + + table = NULL; + /* if the menu item currently has an accelerator then we'll + * remove it before we do anything else. + */ + if (menu_item->accelerator_signal) + { + signame = gtk_signal_name (menu_item->accelerator_signal); + table = gtk_accelerator_table_find (GTK_OBJECT (widget), + signame, + menu_item->accelerator_key, + menu_item->accelerator_mods); + if (!table) + table = GTK_MENU (widget)->accelerator_table; + gtk_widget_remove_accelerator (GTK_WIDGET (menu_item), + table, signame); + } + + if (!table) + table = GTK_MENU (widget)->accelerator_table; + + /* if we aren't simply deleting the accelerator, then we'll install + * the new one now. + */ + if (!delete) + gtk_widget_install_accelerator (GTK_WIDGET (menu_item), + table, "activate", + toupper (event->keyval), + event->state); + + /* check and see if the menu has changed size. */ + gtk_widget_size_request (widget, &widget->requisition); + + allocation.x = widget->allocation.x; + allocation.y = widget->allocation.y; + allocation.width = widget->requisition.width; + allocation.height = widget->requisition.height; + + if ((allocation.width == widget->allocation.width) && + (allocation.height == widget->allocation.height)) + { + gtk_widget_queue_draw (widget); + } + else + { + gtk_widget_size_allocate (GTK_WIDGET (widget), &allocation); + gtk_menu_map (widget); + } + + /* unblock resizes */ + gtk_container_unblock_resize (GTK_CONTAINER (widget)); + } + } + + return FALSE; +} + +static gint +gtk_menu_need_resize (GtkContainer *container) +{ + GtkAllocation allocation; + + g_return_val_if_fail (container != NULL, FALSE); + g_return_val_if_fail (GTK_IS_MENU (container), FALSE); + + if (GTK_WIDGET_VISIBLE (container)) + { + GTK_MENU_SHELL (container)->menu_flag = FALSE; + + gtk_widget_size_request (GTK_WIDGET (container), + >K_WIDGET (container)->requisition); + + allocation.x = GTK_WIDGET (container)->allocation.x; + allocation.y = GTK_WIDGET (container)->allocation.y; + allocation.width = GTK_WIDGET (container)->requisition.width; + allocation.height = GTK_WIDGET (container)->requisition.height; + + gtk_widget_size_allocate (GTK_WIDGET (container), &allocation); + } + else + { + GTK_MENU_SHELL (container)->menu_flag = TRUE; + } + + return FALSE; +} + +static void +gtk_menu_deactivate (GtkMenuShell *menu_shell) +{ + GtkMenuShell *parent; + + g_return_if_fail (menu_shell != NULL); + g_return_if_fail (GTK_IS_MENU (menu_shell)); + + parent = GTK_MENU_SHELL (menu_shell->parent_menu_shell); + + menu_shell->activate_time = 0; + gtk_menu_popdown (GTK_MENU (menu_shell)); + + if (parent) + gtk_menu_shell_deactivate (parent); +} |