• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 
gtk_preserve_window_class_init(GtkPreserveWindowClass * klass)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 
gtk_preserve_window_init(GtkPreserveWindow * widget)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 
gtk_preserve_window_new()66 GtkWidget* gtk_preserve_window_new() {
67   return GTK_WIDGET(g_object_new(GTK_TYPE_PRESERVE_WINDOW, NULL));
68 }
69 
gtk_preserve_window_destroy(GtkObject * object)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 
gtk_preserve_window_realize(GtkWidget * widget)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 
gtk_preserve_window_unrealize(GtkWidget * widget)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 
gtk_preserve_window_get_preserve(GtkPreserveWindow * window)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 
gtk_preserve_window_set_preserve(GtkPreserveWindow * window,gboolean value)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 
gtk_preserve_window_size_allocate(GtkWidget * widget,GtkAllocation * allocation)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 
gtk_preserve_window_delegate_resize(GtkPreserveWindow * widget,gboolean delegate)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 
gtk_preserve_window_set_accessible_factory(GtkPreserveWindow * widget,AtkObject * (* factory)(void * userdata),gpointer userdata)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 
gtk_preserve_window_get_accessible(GtkWidget * widget)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
265