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_native_view_id_manager.h" | |
6 | |
7 #include <gdk/gdkx.h> | |
8 #include <gtk/gtk.h> | |
9 | |
10 #include "base/logging.h" | |
11 #include "base/memory/singleton.h" | |
12 #include "base/rand_util.h" | |
13 #include "ui/gfx/gdk_compat.h" | |
14 #include "ui/gfx/gtk_compat.h" | |
15 #include "ui/gfx/gtk_preserve_window.h" | |
16 | |
17 // ----------------------------------------------------------------------------- | |
18 // Bounce functions for GTK to callback into a C++ object... | |
19 | |
20 void OnRealize(gfx::NativeView widget, void* arg) { | |
21 GtkNativeViewManager* manager = reinterpret_cast<GtkNativeViewManager*>(arg); | |
22 manager->OnRealize(widget); | |
23 } | |
24 | |
25 void OnUnrealize(gfx::NativeView widget, void *arg) { | |
26 GtkNativeViewManager* manager = reinterpret_cast<GtkNativeViewManager*>(arg); | |
27 manager->OnUnrealize(widget); | |
28 } | |
29 | |
30 static void OnDestroy(GtkObject* obj, void* arg) { | |
31 GtkNativeViewManager* manager = reinterpret_cast<GtkNativeViewManager*>(arg); | |
32 manager->OnDestroy(reinterpret_cast<GtkWidget*>(obj)); | |
33 } | |
34 | |
35 // ----------------------------------------------------------------------------- | |
36 | |
37 | |
38 // ----------------------------------------------------------------------------- | |
39 // Public functions... | |
40 | |
41 GtkNativeViewManager::GtkNativeViewManager() { | |
42 } | |
43 | |
44 GtkNativeViewManager::~GtkNativeViewManager() { | |
45 } | |
46 | |
47 // static | |
48 GtkNativeViewManager* GtkNativeViewManager::GetInstance() { | |
49 return Singleton<GtkNativeViewManager>::get(); | |
50 } | |
51 | |
52 gfx::NativeViewId GtkNativeViewManager::GetIdForWidget(gfx::NativeView widget) { | |
53 // This is just for unit tests: | |
54 if (!widget) | |
55 return 0; | |
56 | |
57 base::AutoLock locked(lock_); | |
58 | |
59 std::map<gfx::NativeView, gfx::NativeViewId>::const_iterator i = | |
60 native_view_to_id_.find(widget); | |
61 | |
62 if (i != native_view_to_id_.end()) | |
63 return i->second; | |
64 | |
65 gfx::NativeViewId new_id = | |
66 static_cast<gfx::NativeViewId>(base::RandUint64()); | |
67 while (id_to_info_.find(new_id) != id_to_info_.end()) | |
68 new_id = static_cast<gfx::NativeViewId>(base::RandUint64()); | |
69 | |
70 NativeViewInfo info; | |
71 info.widget = widget; | |
72 if (gtk_widget_get_realized(widget)) { | |
73 GdkWindow *gdk_window = gtk_widget_get_window(widget); | |
74 DCHECK(gdk_window); | |
75 info.x_window_id = GDK_WINDOW_XID(gdk_window); | |
76 } | |
77 | |
78 native_view_to_id_[widget] = new_id; | |
79 id_to_info_[new_id] = info; | |
80 | |
81 g_signal_connect(widget, "realize", G_CALLBACK(::OnRealize), this); | |
82 g_signal_connect(widget, "unrealize", G_CALLBACK(::OnUnrealize), this); | |
83 g_signal_connect(widget, "destroy", G_CALLBACK(::OnDestroy), this); | |
84 | |
85 return new_id; | |
86 } | |
87 | |
88 bool GtkNativeViewManager::GetXIDForId(XID* output, gfx::NativeViewId id) { | |
89 base::AutoLock locked(lock_); | |
90 | |
91 std::map<gfx::NativeViewId, NativeViewInfo>::const_iterator i = | |
92 id_to_info_.find(id); | |
93 | |
94 if (i == id_to_info_.end()) | |
95 return false; | |
96 | |
97 *output = i->second.x_window_id; | |
98 return true; | |
99 } | |
100 | |
101 bool GtkNativeViewManager::GetNativeViewForId(gfx::NativeView* output, | |
102 gfx::NativeViewId id) { | |
103 base::AutoLock locked(lock_); | |
104 | |
105 std::map<gfx::NativeViewId, NativeViewInfo>::const_iterator i = | |
106 id_to_info_.find(id); | |
107 | |
108 if (i == id_to_info_.end()) | |
109 return false; | |
110 | |
111 *output = i->second.widget; | |
112 return true; | |
113 } | |
114 | |
115 bool GtkNativeViewManager::GetPermanentXIDForId(XID* output, | |
116 gfx::NativeViewId id) { | |
117 base::AutoLock locked(lock_); | |
118 | |
119 std::map<gfx::NativeViewId, NativeViewInfo>::iterator i = | |
120 id_to_info_.find(id); | |
121 | |
122 if (i == id_to_info_.end()) | |
123 return false; | |
124 | |
125 // We only return permanent XIDs for widgets that allow us to guarantee that | |
126 // the XID will not change. | |
127 DCHECK(GTK_IS_PRESERVE_WINDOW(i->second.widget)); | |
128 GtkPreserveWindow* widget = | |
129 reinterpret_cast<GtkPreserveWindow*>(i->second.widget); | |
130 gtk_preserve_window_set_preserve(widget, TRUE); | |
131 | |
132 *output = GDK_WINDOW_XID(gtk_widget_get_window(i->second.widget)); | |
133 | |
134 // Update the reference count on the permanent XID. | |
135 PermanentXIDInfo info; | |
136 info.widget = widget; | |
137 info.ref_count = 1; | |
138 std::pair<std::map<XID, PermanentXIDInfo>::iterator, bool> ret = | |
139 perm_xid_to_info_.insert(std::make_pair(*output, info)); | |
140 | |
141 if (!ret.second) { | |
142 DCHECK(ret.first->second.widget == widget); | |
143 ret.first->second.ref_count++; | |
144 } | |
145 | |
146 return true; | |
147 } | |
148 | |
149 bool GtkNativeViewManager::AddRefPermanentXID(XID xid) { | |
150 base::AutoLock locked(lock_); | |
151 | |
152 std::map<XID, PermanentXIDInfo>::iterator i = | |
153 perm_xid_to_info_.find(xid); | |
154 | |
155 if (i == perm_xid_to_info_.end()) | |
156 return false; | |
157 | |
158 i->second.ref_count++; | |
159 | |
160 return true; | |
161 } | |
162 | |
163 void GtkNativeViewManager::ReleasePermanentXID(XID xid) { | |
164 base::AutoLock locked(lock_); | |
165 | |
166 std::map<XID, PermanentXIDInfo>::iterator i = | |
167 perm_xid_to_info_.find(xid); | |
168 | |
169 if (i == perm_xid_to_info_.end()) | |
170 return; | |
171 | |
172 if (i->second.ref_count > 1) { | |
173 i->second.ref_count--; | |
174 } else { | |
175 if (i->second.widget) { | |
176 gtk_preserve_window_set_preserve(i->second.widget, FALSE); | |
177 } else { | |
178 GdkWindow* window = reinterpret_cast<GdkWindow*>( | |
179 gdk_x11_window_lookup_for_display(gdk_display_get_default(), xid)); | |
180 DCHECK(window); | |
181 gdk_window_destroy(window); | |
182 } | |
183 perm_xid_to_info_.erase(i); | |
184 } | |
185 } | |
186 | |
187 // ----------------------------------------------------------------------------- | |
188 | |
189 | |
190 // ----------------------------------------------------------------------------- | |
191 // Private functions... | |
192 | |
193 gfx::NativeViewId GtkNativeViewManager::GetWidgetId(gfx::NativeView widget) { | |
194 lock_.AssertAcquired(); | |
195 | |
196 std::map<gfx::NativeView, gfx::NativeViewId>::const_iterator i = | |
197 native_view_to_id_.find(widget); | |
198 | |
199 CHECK(i != native_view_to_id_.end()); | |
200 return i->second; | |
201 } | |
202 | |
203 void GtkNativeViewManager::OnRealize(gfx::NativeView widget) { | |
204 base::AutoLock locked(lock_); | |
205 | |
206 const gfx::NativeViewId id = GetWidgetId(widget); | |
207 std::map<gfx::NativeViewId, NativeViewInfo>::iterator i = | |
208 id_to_info_.find(id); | |
209 | |
210 CHECK(i != id_to_info_.end()); | |
211 | |
212 GdkWindow* gdk_window = gtk_widget_get_window(widget); | |
213 CHECK(gdk_window); | |
214 i->second.x_window_id = GDK_WINDOW_XID(gdk_window); | |
215 } | |
216 | |
217 void GtkNativeViewManager::OnUnrealize(gfx::NativeView widget) { | |
218 base::AutoLock locked(lock_); | |
219 | |
220 const gfx::NativeViewId id = GetWidgetId(widget); | |
221 std::map<gfx::NativeViewId, NativeViewInfo>::iterator i = | |
222 id_to_info_.find(id); | |
223 | |
224 CHECK(i != id_to_info_.end()); | |
225 } | |
226 | |
227 void GtkNativeViewManager::OnDestroy(gfx::NativeView widget) { | |
228 base::AutoLock locked(lock_); | |
229 | |
230 std::map<gfx::NativeView, gfx::NativeViewId>::iterator i = | |
231 native_view_to_id_.find(widget); | |
232 CHECK(i != native_view_to_id_.end()); | |
233 | |
234 std::map<gfx::NativeViewId, NativeViewInfo>::iterator j = | |
235 id_to_info_.find(i->second); | |
236 CHECK(j != id_to_info_.end()); | |
237 | |
238 // If the XID is supposed to outlive the widget, mark it | |
239 // in the lookup table. | |
240 if (GTK_IS_PRESERVE_WINDOW(widget) && | |
241 gtk_preserve_window_get_preserve( | |
242 reinterpret_cast<GtkPreserveWindow*>(widget))) { | |
243 std::map<XID, PermanentXIDInfo>::iterator k = | |
244 perm_xid_to_info_.find(GDK_WINDOW_XID(gtk_widget_get_window(widget))); | |
245 | |
246 if (k != perm_xid_to_info_.end()) | |
247 k->second.widget = NULL; | |
248 } | |
249 | |
250 native_view_to_id_.erase(i); | |
251 id_to_info_.erase(j); | |
252 } | |
253 | |
254 // ----------------------------------------------------------------------------- | |
OLD | NEW |