// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "ui/gfx/gtk_preserve_window.h" #include #include #include "ui/base/gtk/gtk_compat.h" G_BEGIN_DECLS #define GTK_PRESERVE_WINDOW_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), \ GTK_TYPE_PRESERVE_WINDOW, \ GtkPreserveWindowPrivate)) typedef struct _GtkPreserveWindowPrivate GtkPreserveWindowPrivate; struct _GtkPreserveWindowPrivate { // If true, don't create/destroy windows on realize/unrealize. gboolean preserve_window; // Whether or not we delegate the resize of the GdkWindow // to someone else. gboolean delegate_resize; }; G_DEFINE_TYPE(GtkPreserveWindow, gtk_preserve_window, GTK_TYPE_FIXED) static void gtk_preserve_window_destroy(GtkObject* object); static void gtk_preserve_window_realize(GtkWidget* widget); static void gtk_preserve_window_unrealize(GtkWidget* widget); static void gtk_preserve_window_size_allocate(GtkWidget* widget, GtkAllocation* allocation); static void gtk_preserve_window_class_init(GtkPreserveWindowClass *klass) { GtkWidgetClass* widget_class = reinterpret_cast(klass); widget_class->realize = gtk_preserve_window_realize; widget_class->unrealize = gtk_preserve_window_unrealize; widget_class->size_allocate = gtk_preserve_window_size_allocate; GtkObjectClass* object_class = reinterpret_cast(klass); object_class->destroy = gtk_preserve_window_destroy; GObjectClass* gobject_class = G_OBJECT_CLASS(klass); g_type_class_add_private(gobject_class, sizeof(GtkPreserveWindowPrivate)); } static void gtk_preserve_window_init(GtkPreserveWindow* widget) { GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); priv->preserve_window = FALSE; // These widgets always have their own window. gtk_widget_set_has_window(GTK_WIDGET(widget), TRUE); } GtkWidget* gtk_preserve_window_new() { return GTK_WIDGET(g_object_new(GTK_TYPE_PRESERVE_WINDOW, NULL)); } static void gtk_preserve_window_destroy(GtkObject* object) { GtkWidget* widget = reinterpret_cast(object); GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); GdkWindow* gdk_window = gtk_widget_get_window(widget); if (gdk_window) { gdk_window_set_user_data(gdk_window, NULL); // If the window is preserved, someone else must destroy it. if (!priv->preserve_window) gdk_window_destroy(gdk_window); gtk_widget_set_window(widget, NULL); } GTK_OBJECT_CLASS(gtk_preserve_window_parent_class)->destroy(object); } static void gtk_preserve_window_realize(GtkWidget* widget) { g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); GdkWindow* gdk_window = gtk_widget_get_window(widget); if (gdk_window) { GtkAllocation allocation; gtk_widget_get_allocation(widget, &allocation); gdk_window_reparent(gdk_window, gtk_widget_get_parent_window(widget), allocation.x, allocation.y); GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); if (!priv->delegate_resize) { gdk_window_resize(gdk_window, allocation.width, allocation.height); } gint event_mask = gtk_widget_get_events(widget); event_mask |= GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK; gdk_window_set_events(gdk_window, (GdkEventMask) event_mask); gdk_window_set_user_data(gdk_window, widget); gtk_widget_set_realized(widget, TRUE); gtk_widget_style_attach(widget); gtk_style_set_background(gtk_widget_get_style(widget), gdk_window, GTK_STATE_NORMAL); } else { GTK_WIDGET_CLASS(gtk_preserve_window_parent_class)->realize(widget); } } static void gtk_preserve_window_unrealize(GtkWidget* widget) { g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); if (priv->preserve_window) { GtkWidgetClass* widget_class = GTK_WIDGET_CLASS(gtk_preserve_window_parent_class); GtkContainerClass* container_class = GTK_CONTAINER_CLASS(gtk_preserve_window_parent_class); if (gtk_widget_get_mapped(widget)) { widget_class->unmap(widget); gtk_widget_set_mapped(widget, FALSE); } // This is the behavior from GtkWidget, inherited by GtkFixed. // It is unclear why we should not call the potentially overridden // unrealize method (via the callback), but doing so causes errors. container_class->forall( GTK_CONTAINER(widget), FALSE, reinterpret_cast(gtk_widget_unrealize), NULL); GdkWindow* gdk_window = gtk_widget_get_window(widget); // TODO(erg): Almost all style handling will need to be overhauled in GTK3. gtk_style_detach(gtk_widget_get_style(widget)); gdk_window_reparent(gdk_window, gdk_get_default_root_window(), 0, 0); gtk_selection_remove_all(widget); gdk_window_set_user_data(gdk_window, NULL); gtk_widget_set_realized(widget, FALSE); } else { GTK_WIDGET_CLASS(gtk_preserve_window_parent_class)->unrealize(widget); } } gboolean gtk_preserve_window_get_preserve(GtkPreserveWindow* window) { g_return_val_if_fail(GTK_IS_PRESERVE_WINDOW(window), FALSE); GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(window); return priv->preserve_window; } void gtk_preserve_window_set_preserve(GtkPreserveWindow* window, gboolean value) { g_return_if_fail(GTK_IS_PRESERVE_WINDOW(window)); GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(window); priv->preserve_window = value; GtkWidget* widget = GTK_WIDGET(window); GdkWindow* gdk_window = gtk_widget_get_window(widget); if (value && !gdk_window) { GdkWindowAttr attributes; gint attributes_mask; // We may not know the width and height, so we rely on the fact // that a size-allocation will resize it later. attributes.width = 1; attributes.height = 1; attributes.window_type = GDK_WINDOW_CHILD; attributes.wclass = GDK_INPUT_OUTPUT; attributes.override_redirect = TRUE; attributes.visual = gtk_widget_get_visual(widget); attributes.colormap = gtk_widget_get_colormap(widget); attributes.event_mask = gtk_widget_get_events(widget); attributes.event_mask |= GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK; attributes_mask = GDK_WA_VISUAL | GDK_WA_COLORMAP | GDK_WA_NOREDIR; gdk_window = gdk_window_new( gdk_get_default_root_window(), &attributes, attributes_mask); gtk_widget_set_window(widget, gdk_window); } else if (!value && gdk_window && !gtk_widget_get_realized(widget)) { gdk_window_destroy(gdk_window); gtk_widget_set_window(widget, NULL); } } void gtk_preserve_window_size_allocate(GtkWidget* widget, GtkAllocation* allocation) { g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); gtk_widget_set_allocation(widget, allocation); if (gtk_widget_get_realized(widget)) { GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); GdkWindow* gdk_window = gtk_widget_get_window(widget); if (priv->delegate_resize) { gdk_window_move(gdk_window, allocation->x, allocation->y); } else { gdk_window_move_resize( gdk_window, allocation->x, allocation->y, allocation->width, allocation->height); } } // Propagate resize to children guint16 border_width = gtk_container_get_border_width(GTK_CONTAINER(widget)); GList *children = GTK_FIXED(widget)->children; while (children) { GtkFixedChild *child = reinterpret_cast(children->data); if (gtk_widget_get_visible(child->widget)) { GtkRequisition child_requisition; gtk_widget_get_child_requisition(child->widget, &child_requisition); GtkAllocation child_allocation; child_allocation.x = child->x + border_width; child_allocation.y = child->y + border_width; child_allocation.width = child_requisition.width; child_allocation.height = child_requisition.height; gtk_widget_size_allocate(child->widget, &child_allocation); } children = children->next; } } void gtk_preserve_window_delegate_resize(GtkPreserveWindow* widget, gboolean delegate) { GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); priv->delegate_resize = delegate; } G_END_DECLS