gtk2/gtk/gtkmenu.c
1997-12-18 02:17:14 +00:00

735 lines
19 KiB
C

/* 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 != table)
{
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),
&GTK_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);
}