| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "ui/gfx/gtk_preserve_window.h" | |
| 6 | |
| 7 #include <gdk/gdk.h> | |
| 8 #include <gtk/gtk.h> | |
| 9 | |
| 10 #include "ui/gfx/gtk_compat.h" | |
| 11 | |
| 12 G_BEGIN_DECLS | |
| 13 | |
| 14 #define GTK_PRESERVE_WINDOW_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), \ | |
| 15 GTK_TYPE_PRESERVE_WINDOW, \ | |
| 16 GtkPreserveWindowPrivate)) | |
| 17 | |
| 18 typedef struct _GtkPreserveWindowPrivate GtkPreserveWindowPrivate; | |
| 19 | |
| 20 struct _GtkPreserveWindowPrivate { | |
| 21 // If true, don't create/destroy windows on realize/unrealize. | |
| 22 gboolean preserve_window; | |
| 23 | |
| 24 // Whether or not we delegate the resize of the GdkWindow | |
| 25 // to someone else. | |
| 26 gboolean delegate_resize; | |
| 27 | |
| 28 // Accessible factory and userdata. | |
| 29 AtkObject* (*accessible_factory)(void* userdata); | |
| 30 void* accessible_factory_userdata; | |
| 31 }; | |
| 32 | |
| 33 G_DEFINE_TYPE(GtkPreserveWindow, gtk_preserve_window, GTK_TYPE_FIXED) | |
| 34 | |
| 35 static void gtk_preserve_window_destroy(GtkObject* object); | |
| 36 static void gtk_preserve_window_realize(GtkWidget* widget); | |
| 37 static void gtk_preserve_window_unrealize(GtkWidget* widget); | |
| 38 static void gtk_preserve_window_size_allocate(GtkWidget* widget, | |
| 39 GtkAllocation* allocation); | |
| 40 static AtkObject* gtk_preserve_window_get_accessible(GtkWidget* widget); | |
| 41 | |
| 42 static void gtk_preserve_window_class_init(GtkPreserveWindowClass *klass) { | |
| 43 GtkWidgetClass* widget_class = reinterpret_cast<GtkWidgetClass*>(klass); | |
| 44 widget_class->realize = gtk_preserve_window_realize; | |
| 45 widget_class->unrealize = gtk_preserve_window_unrealize; | |
| 46 widget_class->size_allocate = gtk_preserve_window_size_allocate; | |
| 47 widget_class->get_accessible = gtk_preserve_window_get_accessible; | |
| 48 | |
| 49 GtkObjectClass* object_class = reinterpret_cast<GtkObjectClass*>(klass); | |
| 50 object_class->destroy = gtk_preserve_window_destroy; | |
| 51 | |
| 52 GObjectClass* gobject_class = G_OBJECT_CLASS(klass); | |
| 53 g_type_class_add_private(gobject_class, sizeof(GtkPreserveWindowPrivate)); | |
| 54 } | |
| 55 | |
| 56 static void gtk_preserve_window_init(GtkPreserveWindow* widget) { | |
| 57 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 58 priv->preserve_window = FALSE; | |
| 59 priv->accessible_factory = NULL; | |
| 60 priv->accessible_factory_userdata = NULL; | |
| 61 | |
| 62 // These widgets always have their own window. | |
| 63 gtk_widget_set_has_window(GTK_WIDGET(widget), TRUE); | |
| 64 } | |
| 65 | |
| 66 GtkWidget* gtk_preserve_window_new() { | |
| 67 return GTK_WIDGET(g_object_new(GTK_TYPE_PRESERVE_WINDOW, NULL)); | |
| 68 } | |
| 69 | |
| 70 static void gtk_preserve_window_destroy(GtkObject* object) { | |
| 71 GtkWidget* widget = reinterpret_cast<GtkWidget*>(object); | |
| 72 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 73 | |
| 74 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
| 75 if (gdk_window) { | |
| 76 gdk_window_set_user_data(gdk_window, NULL); | |
| 77 // If the window is preserved, someone else must destroy it. | |
| 78 if (!priv->preserve_window) | |
| 79 gdk_window_destroy(gdk_window); | |
| 80 gtk_widget_set_window(widget, NULL); | |
| 81 } | |
| 82 | |
| 83 GTK_OBJECT_CLASS(gtk_preserve_window_parent_class)->destroy(object); | |
| 84 } | |
| 85 | |
| 86 static void gtk_preserve_window_realize(GtkWidget* widget) { | |
| 87 g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); | |
| 88 | |
| 89 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
| 90 if (gdk_window) { | |
| 91 GtkAllocation allocation; | |
| 92 gtk_widget_get_allocation(widget, &allocation); | |
| 93 | |
| 94 gdk_window_reparent(gdk_window, | |
| 95 gtk_widget_get_parent_window(widget), | |
| 96 allocation.x, | |
| 97 allocation.y); | |
| 98 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 99 if (!priv->delegate_resize) { | |
| 100 gdk_window_resize(gdk_window, | |
| 101 allocation.width, | |
| 102 allocation.height); | |
| 103 } | |
| 104 gint event_mask = gtk_widget_get_events(widget); | |
| 105 event_mask |= GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK; | |
| 106 gdk_window_set_events(gdk_window, (GdkEventMask) event_mask); | |
| 107 gdk_window_set_user_data(gdk_window, widget); | |
| 108 | |
| 109 gtk_widget_set_realized(widget, TRUE); | |
| 110 | |
| 111 gtk_widget_style_attach(widget); | |
| 112 gtk_style_set_background(gtk_widget_get_style(widget), | |
| 113 gdk_window, GTK_STATE_NORMAL); | |
| 114 } else { | |
| 115 GTK_WIDGET_CLASS(gtk_preserve_window_parent_class)->realize(widget); | |
| 116 } | |
| 117 } | |
| 118 | |
| 119 static void gtk_preserve_window_unrealize(GtkWidget* widget) { | |
| 120 g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); | |
| 121 | |
| 122 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 123 if (priv->preserve_window) { | |
| 124 GtkWidgetClass* widget_class = | |
| 125 GTK_WIDGET_CLASS(gtk_preserve_window_parent_class); | |
| 126 GtkContainerClass* container_class = | |
| 127 GTK_CONTAINER_CLASS(gtk_preserve_window_parent_class); | |
| 128 | |
| 129 if (gtk_widget_get_mapped(widget)) { | |
| 130 widget_class->unmap(widget); | |
| 131 | |
| 132 gtk_widget_set_mapped(widget, FALSE); | |
| 133 } | |
| 134 | |
| 135 // This is the behavior from GtkWidget, inherited by GtkFixed. | |
| 136 // It is unclear why we should not call the potentially overridden | |
| 137 // unrealize method (via the callback), but doing so causes errors. | |
| 138 container_class->forall( | |
| 139 GTK_CONTAINER(widget), FALSE, | |
| 140 reinterpret_cast<GtkCallback>(gtk_widget_unrealize), NULL); | |
| 141 | |
| 142 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
| 143 | |
| 144 // TODO(erg): Almost all style handling will need to be overhauled in GTK3. | |
| 145 gtk_style_detach(gtk_widget_get_style(widget)); | |
| 146 gdk_window_reparent(gdk_window, gdk_get_default_root_window(), 0, 0); | |
| 147 gtk_selection_remove_all(widget); | |
| 148 gdk_window_set_user_data(gdk_window, NULL); | |
| 149 | |
| 150 gtk_widget_set_realized(widget, FALSE); | |
| 151 } else { | |
| 152 GTK_WIDGET_CLASS(gtk_preserve_window_parent_class)->unrealize(widget); | |
| 153 } | |
| 154 } | |
| 155 | |
| 156 gboolean gtk_preserve_window_get_preserve(GtkPreserveWindow* window) { | |
| 157 g_return_val_if_fail(GTK_IS_PRESERVE_WINDOW(window), FALSE); | |
| 158 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(window); | |
| 159 | |
| 160 return priv->preserve_window; | |
| 161 } | |
| 162 | |
| 163 void gtk_preserve_window_set_preserve(GtkPreserveWindow* window, | |
| 164 gboolean value) { | |
| 165 g_return_if_fail(GTK_IS_PRESERVE_WINDOW(window)); | |
| 166 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(window); | |
| 167 priv->preserve_window = value; | |
| 168 | |
| 169 GtkWidget* widget = GTK_WIDGET(window); | |
| 170 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
| 171 if (value && !gdk_window) { | |
| 172 GdkWindowAttr attributes; | |
| 173 gint attributes_mask; | |
| 174 | |
| 175 // We may not know the width and height, so we rely on the fact | |
| 176 // that a size-allocation will resize it later. | |
| 177 attributes.width = 1; | |
| 178 attributes.height = 1; | |
| 179 | |
| 180 attributes.window_type = GDK_WINDOW_CHILD; | |
| 181 attributes.wclass = GDK_INPUT_OUTPUT; | |
| 182 attributes.override_redirect = TRUE; | |
| 183 | |
| 184 attributes.visual = gtk_widget_get_visual(widget); | |
| 185 attributes.colormap = gtk_widget_get_colormap(widget); | |
| 186 | |
| 187 attributes.event_mask = gtk_widget_get_events(widget); | |
| 188 attributes.event_mask |= GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK; | |
| 189 | |
| 190 attributes_mask = GDK_WA_VISUAL | GDK_WA_COLORMAP | GDK_WA_NOREDIR; | |
| 191 gdk_window = gdk_window_new( | |
| 192 gdk_get_default_root_window(), &attributes, attributes_mask); | |
| 193 gtk_widget_set_window(widget, gdk_window); | |
| 194 } else if (!value && gdk_window && !gtk_widget_get_realized(widget)) { | |
| 195 gdk_window_destroy(gdk_window); | |
| 196 gtk_widget_set_window(widget, NULL); | |
| 197 } | |
| 198 } | |
| 199 | |
| 200 void gtk_preserve_window_size_allocate(GtkWidget* widget, | |
| 201 GtkAllocation* allocation) { | |
| 202 g_return_if_fail(GTK_IS_PRESERVE_WINDOW(widget)); | |
| 203 | |
| 204 gtk_widget_set_allocation(widget, allocation); | |
| 205 | |
| 206 if (gtk_widget_get_realized(widget)) { | |
| 207 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 208 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
| 209 if (priv->delegate_resize) { | |
| 210 gdk_window_move(gdk_window, allocation->x, allocation->y); | |
| 211 } else { | |
| 212 gdk_window_move_resize( | |
| 213 gdk_window, allocation->x, allocation->y, | |
| 214 allocation->width, allocation->height); | |
| 215 } | |
| 216 } | |
| 217 | |
| 218 // Propagate resize to children | |
| 219 guint16 border_width = gtk_container_get_border_width(GTK_CONTAINER(widget)); | |
| 220 GList *children = GTK_FIXED(widget)->children; | |
| 221 while (children) { | |
| 222 GtkFixedChild *child = reinterpret_cast<GtkFixedChild*>(children->data); | |
| 223 if (gtk_widget_get_visible(child->widget)) { | |
| 224 GtkRequisition child_requisition; | |
| 225 gtk_widget_get_child_requisition(child->widget, &child_requisition); | |
| 226 | |
| 227 GtkAllocation child_allocation; | |
| 228 child_allocation.x = child->x + border_width; | |
| 229 child_allocation.y = child->y + border_width; | |
| 230 child_allocation.width = child_requisition.width; | |
| 231 child_allocation.height = child_requisition.height; | |
| 232 | |
| 233 gtk_widget_size_allocate(child->widget, &child_allocation); | |
| 234 } | |
| 235 children = children->next; | |
| 236 } | |
| 237 } | |
| 238 | |
| 239 void gtk_preserve_window_delegate_resize(GtkPreserveWindow* widget, | |
| 240 gboolean delegate) { | |
| 241 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 242 priv->delegate_resize = delegate; | |
| 243 } | |
| 244 | |
| 245 void gtk_preserve_window_set_accessible_factory( | |
| 246 GtkPreserveWindow* widget, | |
| 247 AtkObject* (*factory)(void* userdata), | |
| 248 gpointer userdata) { | |
| 249 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 250 priv->accessible_factory = factory; | |
| 251 priv->accessible_factory_userdata = userdata; | |
| 252 } | |
| 253 | |
| 254 AtkObject* gtk_preserve_window_get_accessible(GtkWidget* widget) { | |
| 255 GtkPreserveWindowPrivate* priv = GTK_PRESERVE_WINDOW_GET_PRIVATE(widget); | |
| 256 if (priv->accessible_factory) { | |
| 257 return priv->accessible_factory(priv->accessible_factory_userdata); | |
| 258 } else { | |
| 259 return GTK_WIDGET_CLASS(gtk_preserve_window_parent_class) | |
| 260 ->get_accessible(widget); | |
| 261 } | |
| 262 } | |
| 263 | |
| 264 G_END_DECLS | |
| OLD | NEW |