OLD | NEW |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "base/message_pump_glib_x.h" | 5 #include "base/message_pump_x.h" |
6 | 6 |
7 #include <gdk/gdkx.h> | 7 #include <gdk/gdkx.h> |
8 #if defined(HAVE_XINPUT2) | 8 #if defined(HAVE_XINPUT2) |
9 #include <X11/extensions/XInput2.h> | 9 #include <X11/extensions/XInput2.h> |
10 #else | 10 #else |
11 #include <X11/Xlib.h> | 11 #include <X11/Xlib.h> |
12 #endif | 12 #endif |
13 | 13 |
14 #include "base/message_pump_glib_x_dispatch.h" | |
15 | |
16 namespace { | 14 namespace { |
17 | 15 |
18 gboolean PlaceholderDispatch(GSource* source, | 16 gboolean PlaceholderDispatch(GSource* source, |
19 GSourceFunc cb, | 17 GSourceFunc cb, |
20 gpointer data) { | 18 gpointer data) { |
21 return TRUE; | 19 return TRUE; |
22 } | 20 } |
23 | 21 |
24 } // namespace | 22 } // namespace |
25 | 23 |
26 namespace base { | 24 namespace base { |
27 | 25 |
28 MessagePumpGlibX::MessagePumpGlibX() : base::MessagePumpForUI(), | 26 MessagePumpX::MessagePumpX() : MessagePumpGlib(), |
29 #if defined(HAVE_XINPUT2) | 27 #if defined(HAVE_XINPUT2) |
30 xiopcode_(-1), | 28 xiopcode_(-1), |
31 #endif | 29 #endif |
32 gdksource_(NULL), | 30 gdksource_(NULL), |
33 dispatching_event_(false), | 31 dispatching_event_(false), |
34 capture_x_events_(0), | 32 capture_x_events_(0), |
35 capture_gdk_events_(0) { | 33 capture_gdk_events_(0) { |
36 gdk_window_add_filter(NULL, &GdkEventFilter, this); | 34 gdk_window_add_filter(NULL, &GdkEventFilter, this); |
37 gdk_event_handler_set(&EventDispatcherX, this, NULL); | 35 gdk_event_handler_set(&EventDispatcherX, this, NULL); |
38 | 36 |
39 #if defined(HAVE_XINPUT2) | 37 #if defined(HAVE_XINPUT2) |
40 InitializeXInput2(); | 38 InitializeXInput2(); |
41 #endif | 39 #endif |
42 InitializeEventsToCapture(); | 40 InitializeEventsToCapture(); |
43 } | 41 } |
44 | 42 |
45 MessagePumpGlibX::~MessagePumpGlibX() { | 43 MessagePumpX::~MessagePumpX() { |
46 gdk_window_remove_filter(NULL, &GdkEventFilter, this); | 44 gdk_window_remove_filter(NULL, &GdkEventFilter, this); |
47 | 45 gdk_event_handler_set(reinterpret_cast<GdkEventFunc>(gtk_main_do_event), |
48 // It is not necessary to reset the GDK event handler using | 46 this, NULL); |
49 // gdk_event_handler_set since it's done in the destructor for | |
50 // MessagePumpForUI. | |
51 } | 47 } |
52 | 48 |
53 bool MessagePumpGlibX::ShouldCaptureXEvent(XEvent* xev) { | 49 bool MessagePumpX::ShouldCaptureXEvent(XEvent* xev) { |
54 return capture_x_events_[xev->type] | 50 return capture_x_events_[xev->type] |
55 #if defined(HAVE_XINPUT2) | 51 #if defined(HAVE_XINPUT2) |
56 && (xev->type != GenericEvent || xev->xcookie.extension == xiopcode_) | 52 && (xev->type != GenericEvent || xev->xcookie.extension == xiopcode_) |
57 #endif | 53 #endif |
58 ; | 54 ; |
59 } | 55 } |
60 | 56 |
61 | 57 |
62 bool MessagePumpGlibX::ProcessXEvent(XEvent* xev) { | 58 bool MessagePumpX::ProcessXEvent(XEvent* xev) { |
63 bool should_quit = false; | 59 bool should_quit = false; |
64 | 60 |
65 #if defined(HAVE_XINPUT2) | 61 #if defined(HAVE_XINPUT2) |
66 bool have_cookie = false; | 62 bool have_cookie = false; |
67 if (xev->type == GenericEvent && | 63 if (xev->type == GenericEvent && |
68 XGetEventData(xev->xgeneric.display, &xev->xcookie)) { | 64 XGetEventData(xev->xgeneric.display, &xev->xcookie)) { |
69 have_cookie = true; | 65 have_cookie = true; |
70 } | 66 } |
71 #endif | 67 #endif |
72 | 68 |
73 if (!WillProcessXEvent(xev)) { | 69 if (!WillProcessXEvent(xev)) { |
74 MessagePumpGlibXDispatcher::DispatchStatus status = | 70 MessagePumpDispatcher::DispatchStatus status = |
75 static_cast<MessagePumpGlibXDispatcher*> | 71 GetDispatcher()->DispatchX(xev); |
76 (GetDispatcher())->DispatchX(xev); | |
77 | 72 |
78 if (status == MessagePumpGlibXDispatcher::EVENT_QUIT) { | 73 if (status == MessagePumpDispatcher::EVENT_QUIT) { |
79 should_quit = true; | 74 should_quit = true; |
80 Quit(); | 75 Quit(); |
81 } else if (status == MessagePumpGlibXDispatcher::EVENT_IGNORED) { | 76 } else if (status == MessagePumpDispatcher::EVENT_IGNORED) { |
82 VLOG(1) << "Event (" << xev->type << ") not handled."; | 77 VLOG(1) << "Event (" << xev->type << ") not handled."; |
83 } | 78 } |
84 } | 79 } |
85 | 80 |
86 #if defined(HAVE_XINPUT2) | 81 #if defined(HAVE_XINPUT2) |
87 if (have_cookie) { | 82 if (have_cookie) { |
88 XFreeEventData(xev->xgeneric.display, &xev->xcookie); | 83 XFreeEventData(xev->xgeneric.display, &xev->xcookie); |
89 } | 84 } |
90 #endif | 85 #endif |
91 | 86 |
92 return should_quit; | 87 return should_quit; |
93 } | 88 } |
94 | 89 |
95 bool MessagePumpGlibX::RunOnce(GMainContext* context, bool block) { | 90 bool MessagePumpX::RunOnce(GMainContext* context, bool block) { |
96 GdkDisplay* gdisp = gdk_display_get_default(); | 91 GdkDisplay* gdisp = gdk_display_get_default(); |
97 if (!gdisp || !GetDispatcher()) | 92 if (!gdisp || !GetDispatcher()) |
98 return MessagePumpForUI::RunOnce(context, block); | 93 return g_main_context_iteration(context, block); |
99 | 94 |
100 Display* display = GDK_DISPLAY_XDISPLAY(gdisp); | 95 Display* display = GDK_DISPLAY_XDISPLAY(gdisp); |
101 | 96 |
102 if (XPending(display)) { | 97 if (XPending(display)) { |
103 XEvent xev; | 98 XEvent xev; |
104 XPeekEvent(display, &xev); | 99 XPeekEvent(display, &xev); |
105 | 100 |
106 if (ShouldCaptureXEvent(&xev)) { | 101 if (ShouldCaptureXEvent(&xev)) { |
107 XNextEvent(display, &xev); | 102 XNextEvent(display, &xev); |
108 if (ProcessXEvent(&xev)) | 103 if (ProcessXEvent(&xev)) |
(...skipping 21 matching lines...) Expand all Loading... |
130 dispatching_event_ = false; | 125 dispatching_event_ = false; |
131 | 126 |
132 gdksource_->source_funcs->dispatch = cb; | 127 gdksource_->source_funcs->dispatch = cb; |
133 } else { | 128 } else { |
134 retvalue = g_main_context_iteration(context, block); | 129 retvalue = g_main_context_iteration(context, block); |
135 } | 130 } |
136 | 131 |
137 return retvalue; | 132 return retvalue; |
138 } | 133 } |
139 | 134 |
140 GdkFilterReturn MessagePumpGlibX::GdkEventFilter(GdkXEvent* gxevent, | 135 GdkFilterReturn MessagePumpX::GdkEventFilter(GdkXEvent* gxevent, |
141 GdkEvent* gevent, | 136 GdkEvent* gevent, |
142 gpointer data) { | 137 gpointer data) { |
143 MessagePumpGlibX* pump = static_cast<MessagePumpGlibX*>(data); | 138 MessagePumpX* pump = static_cast<MessagePumpX*>(data); |
144 XEvent* xev = static_cast<XEvent*>(gxevent); | 139 XEvent* xev = static_cast<XEvent*>(gxevent); |
145 | 140 |
146 if (pump->ShouldCaptureXEvent(xev) && pump->GetDispatcher()) { | 141 if (pump->ShouldCaptureXEvent(xev) && pump->GetDispatcher()) { |
147 pump->ProcessXEvent(xev); | 142 pump->ProcessXEvent(xev); |
148 return GDK_FILTER_REMOVE; | 143 return GDK_FILTER_REMOVE; |
149 } | 144 } |
150 | 145 |
151 return GDK_FILTER_CONTINUE; | 146 return GDK_FILTER_CONTINUE; |
152 } | 147 } |
153 | 148 |
154 bool MessagePumpGlibX::WillProcessXEvent(XEvent* xevent) { | 149 bool MessagePumpX::WillProcessXEvent(XEvent* xevent) { |
155 ObserverListBase<Observer>::Iterator it(observers()); | 150 ObserverListBase<MessagePumpObserver>::Iterator it(observers()); |
156 Observer* obs; | 151 MessagePumpObserver* obs; |
157 while ((obs = it.GetNext()) != NULL) { | 152 while ((obs = it.GetNext()) != NULL) { |
158 MessagePumpXObserver* xobs = | 153 if (obs->WillProcessXEvent(xevent)) |
159 static_cast<MessagePumpXObserver*>(obs); | |
160 if (xobs->WillProcessXEvent(xevent)) | |
161 return true; | 154 return true; |
162 } | 155 } |
163 return false; | 156 return false; |
164 } | 157 } |
165 | 158 |
166 void MessagePumpGlibX::EventDispatcherX(GdkEvent* event, gpointer data) { | 159 void MessagePumpX::EventDispatcherX(GdkEvent* event, gpointer data) { |
167 MessagePumpGlibX* pump_x = reinterpret_cast<MessagePumpGlibX*>(data); | 160 MessagePumpX* pump_x = reinterpret_cast<MessagePumpX*>(data); |
168 | 161 |
169 if (!pump_x->gdksource_) { | 162 if (!pump_x->gdksource_) { |
170 pump_x->gdksource_ = g_main_current_source(); | 163 pump_x->gdksource_ = g_main_current_source(); |
171 if (pump_x->gdksource_) | 164 if (pump_x->gdksource_) |
172 pump_x->gdkdispatcher_ = pump_x->gdksource_->source_funcs->dispatch; | 165 pump_x->gdkdispatcher_ = pump_x->gdksource_->source_funcs->dispatch; |
173 } else if (!pump_x->IsDispatchingEvent()) { | 166 } else if (!pump_x->IsDispatchingEvent()) { |
174 if (event->type != GDK_NOTHING && | 167 if (event->type != GDK_NOTHING && |
175 pump_x->capture_gdk_events_[event->type]) { | 168 pump_x->capture_gdk_events_[event->type]) { |
176 NOTREACHED() << "GDK received an event it shouldn't have"; | 169 NOTREACHED() << "GDK received an event it shouldn't have"; |
177 } | 170 } |
178 } | 171 } |
179 | 172 |
180 pump_x->DispatchEvents(event); | 173 gtk_main_do_event(event); |
181 } | 174 } |
182 | 175 |
183 void MessagePumpGlibX::InitializeEventsToCapture(void) { | 176 void MessagePumpX::InitializeEventsToCapture(void) { |
184 // TODO(sad): Decide which events we want to capture and update the tables | 177 // TODO(sad): Decide which events we want to capture and update the tables |
185 // accordingly. | 178 // accordingly. |
186 capture_x_events_[KeyPress] = true; | 179 capture_x_events_[KeyPress] = true; |
187 capture_gdk_events_[GDK_KEY_PRESS] = true; | 180 capture_gdk_events_[GDK_KEY_PRESS] = true; |
188 | 181 |
189 capture_x_events_[KeyRelease] = true; | 182 capture_x_events_[KeyRelease] = true; |
190 capture_gdk_events_[GDK_KEY_RELEASE] = true; | 183 capture_gdk_events_[GDK_KEY_RELEASE] = true; |
191 | 184 |
192 capture_x_events_[ButtonPress] = true; | 185 capture_x_events_[ButtonPress] = true; |
193 capture_gdk_events_[GDK_BUTTON_PRESS] = true; | 186 capture_gdk_events_[GDK_BUTTON_PRESS] = true; |
194 | 187 |
195 capture_x_events_[ButtonRelease] = true; | 188 capture_x_events_[ButtonRelease] = true; |
196 capture_gdk_events_[GDK_BUTTON_RELEASE] = true; | 189 capture_gdk_events_[GDK_BUTTON_RELEASE] = true; |
197 | 190 |
198 capture_x_events_[MotionNotify] = true; | 191 capture_x_events_[MotionNotify] = true; |
199 capture_gdk_events_[GDK_MOTION_NOTIFY] = true; | 192 capture_gdk_events_[GDK_MOTION_NOTIFY] = true; |
200 | 193 |
201 #if defined(HAVE_XINPUT2) | 194 #if defined(HAVE_XINPUT2) |
202 capture_x_events_[GenericEvent] = true; | 195 capture_x_events_[GenericEvent] = true; |
203 #endif | 196 #endif |
204 } | 197 } |
205 | 198 |
206 #if defined(HAVE_XINPUT2) | 199 #if defined(HAVE_XINPUT2) |
207 void MessagePumpGlibX::InitializeXInput2(void) { | 200 void MessagePumpX::InitializeXInput2(void) { |
208 GdkDisplay* display = gdk_display_get_default(); | 201 GdkDisplay* display = gdk_display_get_default(); |
209 if (!display) | 202 if (!display) |
210 return; | 203 return; |
211 | 204 |
212 Display* xdisplay = GDK_DISPLAY_XDISPLAY(display); | 205 Display* xdisplay = GDK_DISPLAY_XDISPLAY(display); |
213 int event, err; | 206 int event, err; |
214 | 207 |
215 if (!XQueryExtension(xdisplay, "XInputExtension", &xiopcode_, &event, &err)) { | 208 if (!XQueryExtension(xdisplay, "XInputExtension", &xiopcode_, &event, &err)) { |
216 VLOG(1) << "X Input extension not available."; | 209 VLOG(1) << "X Input extension not available."; |
217 xiopcode_ = -1; | 210 xiopcode_ = -1; |
218 return; | 211 return; |
219 } | 212 } |
220 | 213 |
221 int major = 2, minor = 0; | 214 int major = 2, minor = 0; |
222 if (XIQueryVersion(xdisplay, &major, &minor) == BadRequest) { | 215 if (XIQueryVersion(xdisplay, &major, &minor) == BadRequest) { |
223 VLOG(1) << "XInput2 not supported in the server."; | 216 VLOG(1) << "XInput2 not supported in the server."; |
224 xiopcode_ = -1; | 217 xiopcode_ = -1; |
225 return; | 218 return; |
226 } | 219 } |
227 } | 220 } |
228 #endif // HAVE_XINPUT2 | 221 #endif // HAVE_XINPUT2 |
229 | 222 |
230 bool MessagePumpXObserver::WillProcessXEvent(XEvent* xev) { | 223 bool MessagePumpObserver::WillProcessXEvent(XEvent* xev) { |
231 return false; | 224 return false; |
232 } | 225 } |
233 | 226 |
234 } // namespace base | 227 } // namespace base |
OLD | NEW |