summaryrefslogtreecommitdiffstats
path: root/x11-toolkits/gtk20/files/gtk-shadow-patch
diff options
context:
space:
mode:
Diffstat (limited to 'x11-toolkits/gtk20/files/gtk-shadow-patch')
-rw-r--r--x11-toolkits/gtk20/files/gtk-shadow-patch594
1 files changed, 594 insertions, 0 deletions
diff --git a/x11-toolkits/gtk20/files/gtk-shadow-patch b/x11-toolkits/gtk20/files/gtk-shadow-patch
new file mode 100644
index 000000000..a71838b03
--- /dev/null
+++ b/x11-toolkits/gtk20/files/gtk-shadow-patch
@@ -0,0 +1,594 @@
+--- gtk/gtkmenu.c.orig Mon Jun 27 13:36:34 2005
++++ gtk/gtkmenu.c Mon Jul 4 22:18:27 2005
+@@ -51,6 +51,7 @@
+
+ #define DEFAULT_POPUP_DELAY 225
+ #define DEFAULT_POPDOWN_DELAY 1000
++#define DEFAULT_SHADOW_DELAY 50
+
+ #define NAVIGATION_REGION_OVERSHOOT 50 /* How much the navigation region
+ * extends below the submenu
+@@ -93,6 +94,11 @@
+ gboolean have_layout;
+ gint n_rows;
+ gint n_columns;
++
++ /* Shadow patch addon */
++ GdkPixbuf *east, *south;
++ GdkWindow *east_shadow, *south_shadow;
++ guint32 timeout_id;
+ };
+
+ typedef struct
+@@ -126,6 +132,60 @@
+ CHILD_PROP_BOTTOM_ATTACH
+ };
+
++enum side {
++ EAST_SIDE,
++ SOUTH_SIDE
++};
++
++const double shadow_strip_l[5] = {
++ .937, .831, .670, .478, .180
++};
++
++const double bottom_left_corner[25] = {
++ 1.00, .682, .423, .333, .258,
++ 1.00, .898, .800, .682, .584,
++ 1.00, .937, .874, .800, .737,
++ 1.00, .968, .937, .898, .866,
++ 1.00, .988, .976, .960, .945
++};
++
++const double bottom_right_corner[25] = {
++ .258, .584, .737, .866, .945,
++ .584, .682, .800, .898, .960,
++ .737, .800, .874, .937, .976,
++ .866, .898, .937, .968, .988,
++ .945, .960, .976, .988, .996
++};
++
++const double top_right_corner[25] = {
++ 1.00, 1.00, 1.00, 1.00, 1.00,
++ .686, .898, .937, .968, .988,
++ .423, .803, .874, .937, .976,
++ .333, .686, .800, .898, .960,
++ .258, .584, .737, .866, .945
++};
++
++const double top_left_corner[25] = {
++ .988, .968, .937, .898, .498,
++ .976, .937, .874, .803, .423,
++ .960, .898, .800, .686, .333,
++ .945, .866, .737, .584, .258,
++ .941, .847, .698, .521, .215
++};
++
++static GdkPixbuf *get_pixbuf (GtkMenu *menu,
++ int x,
++ int y,
++ int width,
++ int height);
++static void shadow_paint (GtkWidget *widget,
++ GdkRectangle *area,
++ enum side shadow);
++static void pixbuf_add_shadow (GdkPixbuf *pb,
++ enum side shadow);
++static gboolean map_shadow_windows (gpointer data);
++static void shadow_add_timeout (GtkWidget *widget);
++static void shadow_remove_timeout (GtkWidget *widget);
+ static void gtk_menu_class_init (GtkMenuClass *klass);
+ static void gtk_menu_init (GtkMenu *menu);
+ static void gtk_menu_set_property (GObject *object,
+@@ -246,6 +306,9 @@
+ {
+ GtkMenuPrivate *priv = (GtkMenuPrivate *)data;
+
++ if (priv->timeout_id > 0)
++ g_source_remove (priv->timeout_id);
++
+ g_free (priv->heights);
+
+ g_free (priv);
+@@ -466,6 +529,307 @@
+
+ }
+
++static GdkPixbuf *
++get_pixbuf (GtkMenu *menu,
++ int x,
++ int y,
++ int width,
++ int height)
++{
++ GdkPixbuf *dest, *src;
++ GdkScreen *screen = gtk_widget_get_screen (GTK_WIDGET(menu));
++ GdkWindow *root = gdk_screen_get_root_window (screen);
++ gint screen_height = gdk_screen_get_height (screen);
++ gint screen_width = gdk_screen_get_width (screen);
++ gint original_width = width;
++ gint original_height = height;
++
++ if (x < 0)
++ {
++ width += x;
++ x = 0;
++ }
++
++ if (y < 0)
++ {
++ height += y;
++ y = 0;
++ }
++
++ if (x + width > screen_width)
++ {
++ width = screen_width - x;
++ }
++
++ if (y + height > screen_height)
++ {
++ height = screen_height - y;
++ }
++
++ if (width <= 0 || height <= 0)
++ return NULL;
++
++ dest = gdk_pixbuf_new (GDK_COLORSPACE_RGB, FALSE, 8,
++ original_width, original_height);
++ src = gdk_pixbuf_get_from_drawable (NULL, root, NULL, x, y, 0, 0,
++ width, height);
++ gdk_pixbuf_copy_area (src, 0, 0, width, height, dest, 0, 0);
++
++ g_object_unref (G_OBJECT (src));
++
++ return dest;
++}
++
++static void
++shadow_paint(GtkWidget *widget, GdkRectangle *area, enum side shadow)
++{
++ GtkMenu *menu = GTK_MENU (widget);
++ GtkMenuPrivate *private = gtk_menu_get_private (menu);
++ gint width, height;
++ GdkGC *gc = widget->style->black_gc;
++
++ switch (shadow)
++ {
++ case EAST_SIDE:
++ if (private->east != NULL)
++ {
++ if (area)
++ gdk_gc_set_clip_rectangle (gc, area);
++
++ width = gdk_pixbuf_get_width (private->east);
++ height = gdk_pixbuf_get_height (private->east);
++
++ gdk_draw_pixbuf (private->east_shadow, gc, private->east, 0, 0, 0, 0,
++ width, height, GDK_RGB_DITHER_NONE, 0, 0);
++
++ if (area)
++ gdk_gc_set_clip_rectangle (gc, NULL);
++ }
++ break;
++ case SOUTH_SIDE:
++ if (private->south != NULL)
++ {
++ if (area)
++ gdk_gc_set_clip_rectangle (gc, area);
++
++ width = gdk_pixbuf_get_width (private->south);
++ height = gdk_pixbuf_get_height (private->south);
++
++ gdk_draw_pixbuf (private->south_shadow, gc, private->south, 0, 0, 0, 0,
++ width, height, GDK_RGB_DITHER_NONE, 0, 0);
++
++ if (area)
++ gdk_gc_set_clip_rectangle (gc, NULL);
++ }
++ break;
++ default:
++ break;
++ }
++}
++
++static void
++pixbuf_add_shadow (GdkPixbuf *pb,
++ enum side shadow)
++{
++ gint width, rowstride, height;
++ gint i;
++ guchar *pixels, *p;
++
++ width = gdk_pixbuf_get_width (pb);
++ height = gdk_pixbuf_get_height (pb);
++ rowstride = gdk_pixbuf_get_rowstride (pb);
++ pixels = gdk_pixbuf_get_pixels (pb);
++
++ switch (shadow)
++ {
++ case EAST_SIDE:
++ if (height > 5)
++ {
++ for (i = 0; i < width; i++)
++ {
++ gint j, k;
++
++ p = pixels + (i * rowstride);
++ for (j = 0, k = 0; j < 3 * width; j += 3, k++)
++ {
++ p[j] = (guchar) (p[j] * top_right_corner [i * width + k]);
++ p[j + 1] = (guchar) (p[j + 1] * top_right_corner [i * width + k]);
++ p[j + 2] = (guchar) (p[j + 2] * top_right_corner [i * width + k]);
++ }
++ }
++
++ i = 5;
++ }
++ else
++ {
++ i = 0;
++ }
++
++ for (;i < height; i++)
++ {
++ gint j, k;
++
++ p = pixels + (i * rowstride);
++ for (j = 0, k = 0; j < 3 * width; j += 3, k++)
++ {
++ p[j] = (guchar) (p[j] * shadow_strip_l[width - 1 - k]);
++ p[j + 1] = (guchar) (p[j + 1] * shadow_strip_l[width - 1 - k]);
++ p[j + 2] = (guchar) (p[j + 2] * shadow_strip_l[width - 1 - k]);
++ }
++ }
++ break;
++
++ case SOUTH_SIDE:
++ for (i = 0; i < height; i++)
++ {
++ gint j, k;
++
++ p = pixels + (i * rowstride);
++ for (j = 0, k = 0; j < 3 * height; j += 3, k++)
++ {
++
++ p[j] = (guchar) (p[j] * bottom_left_corner[i * height + k]);
++ p[j + 1] = (guchar) (p[j + 1] * bottom_left_corner[i * height + k]);
++ p[j + 2] = (guchar) (p[j + 2] * bottom_left_corner[i * height + k]);
++ }
++
++ p = pixels + (i * rowstride) + 3 * height;
++ for (j = 0, k = 0; j < (width * 3) - (6 * height); j += 3, k++)
++ {
++ p[j] = (guchar) (p[j] * bottom_right_corner [i * height]);
++ p[j + 1] = (guchar) (p[j + 1] * bottom_right_corner [i * height]);
++ p[j + 2] = (guchar) (p[j + 2] * bottom_right_corner [i * height]);
++ }
++
++ p = pixels + (i * rowstride) + ((width * 3) - (3 * height));
++ for (j = 0, k = 0; j < 3 * height; j += 3, k++)
++ {
++ p[j] = (guchar) (p[j] * bottom_right_corner[i * height + k]);
++ p[j + 1] = (guchar) (p[j + 1] * bottom_right_corner[i * height + k]);
++ p[j + 2] = (guchar) (p[j + 2] * bottom_right_corner[i * height + k]);
++ }
++ }
++ break;
++
++ default:
++ break;
++ }
++}
++
++static gboolean
++map_shadow_windows (gpointer data)
++{
++ GtkMenu *menu = GTK_MENU (data);
++ GtkMenuPrivate *private = gtk_menu_get_private (menu);
++ GtkWidget *widget = GTK_WIDGET (data);
++ GdkPixbuf *pixbuf;
++
++ pixbuf = get_pixbuf (menu,
++ private->x + widget->allocation.width, private->y,
++ 5, widget->allocation.height);
++ if (pixbuf != NULL)
++ {
++ pixbuf_add_shadow (pixbuf, EAST_SIDE);
++ if (private->east != NULL)
++ {
++ g_object_unref (G_OBJECT (private->east));
++ }
++ private->east = pixbuf;
++ }
++
++ pixbuf = get_pixbuf (menu,
++ private->x, private->y + widget->allocation.height,
++ widget->allocation.width + 5, 5);
++ if (pixbuf != NULL)
++ {
++ pixbuf_add_shadow (pixbuf, SOUTH_SIDE);
++ if (private->south != NULL)
++ {
++ g_object_unref (G_OBJECT (private->south));
++ }
++ private->south = pixbuf;
++ }
++
++ gdk_window_move_resize (private->east_shadow,
++ private->x + widget->allocation.width, private->y,
++ 5, widget->allocation.height);
++
++ gdk_window_move_resize (private->south_shadow,
++ private->x, private->y + widget->allocation.height,
++ widget->allocation.width + 5, 5);
++
++ gdk_window_show (private->east_shadow);
++ gdk_window_show (private->south_shadow);
++
++ shadow_paint(widget, NULL, EAST_SIDE);
++ shadow_paint(widget, NULL, SOUTH_SIDE);
++
++ private->timeout_id = 0;
++ return FALSE;
++}
++
++static void
++shadow_add_timeout(GtkWidget *widget)
++{
++ GtkMenuPrivate *private = gtk_menu_get_private (GTK_MENU (widget));
++ gboolean menu_shadow;
++ gint shadow_delay;
++
++ if (private->have_position)
++ {
++ g_object_get (G_OBJECT (gtk_widget_get_settings (widget)),
++ "gtk-menu-drop-shadow", &menu_shadow, NULL);
++
++ if (menu_shadow)
++ {
++ if (private->timeout_id > 0)
++ {
++ g_source_remove (private->timeout_id);
++ }
++
++
++ g_object_get (G_OBJECT (gtk_widget_get_settings (widget)),
++ "gtk-menu-shadow-delay", &shadow_delay,
++ NULL);
++
++ private->timeout_id = g_timeout_add (shadow_delay, map_shadow_windows, widget);
++ }
++ }
++}
++
++static void
++shadow_remove_timeout (GtkWidget *widget)
++{
++ GtkMenu *menu = GTK_MENU (widget);
++ GtkMenuPrivate *private = gtk_menu_get_private (menu);
++
++ if (private->timeout_id > 0)
++ {
++ g_source_remove (private->timeout_id);
++ private->timeout_id = 0;
++ }
++ else
++ {
++ if (private->east_shadow)
++ gdk_window_hide (private->east_shadow);
++
++ if (private->south_shadow)
++ gdk_window_hide (private->south_shadow);
++
++ if (private->east)
++ {
++ g_object_unref (G_OBJECT (private->east));
++ private->east = NULL;
++ }
++
++ if (private->south)
++ {
++ g_object_unref (G_OBJECT (private->south));
++ private->south = NULL;
++ }
++ }
++}
++
+ static void
+ gtk_menu_class_init (GtkMenuClass *class)
+ {
+@@ -713,6 +1077,20 @@
+ DEFAULT_POPDOWN_DELAY,
+ GTK_PARAM_READWRITE));
+
++ gtk_settings_install_property (g_param_spec_boolean ("gtk-menu-drop-shadow",
++ _("Display menu drop-shadow"),
++ _("Whether menu drop-shadow should be displayed"),
++ TRUE,
++ G_PARAM_READWRITE));
++
++ gtk_settings_install_property (g_param_spec_int ("gtk-menu-shadow-delay",
++ _("Delay before drop-shadow appear"),
++ _("Minimum time before drop-shadow appear under the menu"),
++ 0,
++ G_MAXINT,
++ DEFAULT_SHADOW_DELAY,
++ G_PARAM_READWRITE));
++
+ }
+
+
+@@ -925,6 +1303,15 @@
+ menu->lower_arrow_prelight = FALSE;
+
+ priv->have_layout = FALSE;
++
++ /* Shadow patch */
++ priv->east_shadow = NULL;
++ priv->south_shadow = NULL;
++
++ priv->east = NULL;
++ priv->south = NULL;
++
++ priv->timeout_id = 0;
+ }
+
+ static void
+@@ -997,6 +1384,7 @@
+ return;
+ }
+
++ shadow_remove_timeout(GTK_WIDGET(menu));
+ if (menu->torn_off)
+ {
+ gtk_window_set_screen (GTK_WINDOW (menu->tearoff_window), new_screen);
+@@ -1448,6 +1836,7 @@
+
+ if (xgrab_shell == widget)
+ popup_grab_on_window (widget->window, activate_time, grab_keyboard); /* Should always succeed */
++ shadow_add_timeout(GTK_WIDGET (menu));
+ gtk_grab_add (GTK_WIDGET (menu));
+ }
+
+@@ -1458,7 +1847,7 @@
+ GtkMenuShell *menu_shell;
+
+ g_return_if_fail (GTK_IS_MENU (menu));
+-
++
+ menu_shell = GTK_MENU_SHELL (menu);
+ private = gtk_menu_get_private (menu);
+
+@@ -1523,6 +1912,7 @@
+ menu_shell->have_xgrab = FALSE;
+ gtk_grab_remove (GTK_WIDGET (menu));
+
++ shadow_remove_timeout(GTK_WIDGET (menu));
+ menu_grab_transfer_window_destroy (menu);
+ }
+
+@@ -1990,10 +2380,16 @@
+ if (GTK_WIDGET_REALIZED (widget))
+ {
+ GtkMenu *menu = GTK_MENU (widget);
++ GtkMenuPrivate *private;
+
++ private = gtk_menu_get_private (menu);
++
+ gtk_style_set_background (widget->style, menu->bin_window, GTK_STATE_NORMAL);
+ gtk_style_set_background (widget->style, menu->view_window, GTK_STATE_NORMAL);
+ gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
++
++ gdk_window_set_back_pixmap (private->east_shadow, NULL, FALSE);
++ gdk_window_set_back_pixmap (private->south_shadow, NULL, FALSE);
+ }
+ }
+
+@@ -2004,6 +2400,7 @@
+ gint attributes_mask;
+ gint border_width;
+ GtkMenu *menu;
++ GtkMenuPrivate *private;
+ GtkWidget *child;
+ GList *children;
+ guint vertical_padding;
+@@ -2011,6 +2408,7 @@
+ g_return_if_fail (GTK_IS_MENU (widget));
+
+ menu = GTK_MENU (widget);
++ private = gtk_menu_get_private (menu);
+
+ GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
+
+@@ -2081,6 +2479,25 @@
+
+ gdk_window_show (menu->bin_window);
+ gdk_window_show (menu->view_window);
++
++ /* Drop shadow */
++
++ attributes.window_type = GDK_WINDOW_TEMP;
++ attributes.override_redirect = TRUE;
++
++ attributes_mask = GDK_WA_NOREDIR | GDK_WA_VISUAL | GDK_WA_COLORMAP;
++
++ /* East drop shadow */
++ private->east_shadow = gdk_window_new (gtk_widget_get_root_window (widget),
++ &attributes, attributes_mask);
++ gdk_window_set_user_data (private->east_shadow, menu);
++ gdk_window_set_back_pixmap (private->east_shadow, NULL, FALSE);
++
++ /* South drop shadow */
++ private->south_shadow = gdk_window_new (gtk_widget_get_root_window (widget),
++ &attributes, attributes_mask);
++ gdk_window_set_user_data (private->south_shadow, menu);
++ gdk_window_set_back_pixmap (private->south_shadow, NULL, FALSE);
+ }
+
+ static gboolean
+@@ -2143,10 +2560,12 @@
+ gtk_menu_unrealize (GtkWidget *widget)
+ {
+ GtkMenu *menu;
++ GtkMenuPrivate *private;
+
+ g_return_if_fail (GTK_IS_MENU (widget));
+
+ menu = GTK_MENU (widget);
++ private = gtk_menu_get_private (menu);
+
+ menu_grab_transfer_window_destroy (menu);
+
+@@ -2158,6 +2577,15 @@
+ gdk_window_destroy (menu->bin_window);
+ menu->bin_window = NULL;
+
++ /* Shadows */
++ gdk_window_set_user_data (private->east_shadow, NULL);
++ gdk_window_destroy (private->east_shadow);
++ private->east_shadow = NULL;
++
++ gdk_window_set_user_data (private->south_shadow, NULL);
++ gdk_window_destroy (private->south_shadow);
++ private->south_shadow = NULL;
++
+ (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget);
+ }
+
+@@ -2312,8 +2740,15 @@
+ y,
+ width,
+ height);
+- }
+
++ if (GTK_WIDGET_MAPPED (widget))
++ {
++ /* Remap the shadows as the menu size has changed */
++ shadow_remove_timeout(widget);
++ shadow_add_timeout(widget);
++ }
++ }
++
+ if (menu_shell->children)
+ {
+ gint base_width = width / gtk_menu_get_n_columns (menu);
+@@ -2410,7 +2845,7 @@
+ }
+ }
+ }
+- }
++ }
+ }
+
+ static void
+@@ -3048,7 +3483,7 @@
+
+ gtk_menu_stop_scrolling (menu);
+ }
+- }
++ }
+ }
+
+ static gboolean