| Index: base/message_pump_glib_x.cc
|
| diff --git a/base/message_pump_glib_x.cc b/base/message_pump_glib_x.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..f77f61d10c3adedd9028e3837703cc0e29a530be
|
| --- /dev/null
|
| +++ b/base/message_pump_glib_x.cc
|
| @@ -0,0 +1,104 @@
|
| +// Copyright (c) 2010 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 "base/message_pump_glib_x.h"
|
| +
|
| +#include <gdk/gdkx.h>
|
| +#include <X11/Xlib.h>
|
| +
|
| +namespace {
|
| +
|
| +gboolean PlaceholderDispatch(GSource* source,
|
| + GSourceFunc cb,
|
| + gpointer data) {
|
| + return TRUE;
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +namespace base {
|
| +
|
| +MessagePumpGlibX::MessagePumpGlibX() : base::MessagePumpForUI(),
|
| + gdksource_(NULL),
|
| + dispatching_event_(false),
|
| + capture_x_events_(0),
|
| + capture_gdk_events_(0) {
|
| + gdk_event_handler_set(&EventDispatcherX, this, NULL);
|
| +
|
| + InitializeEventsToCapture();
|
| +}
|
| +
|
| +MessagePumpGlibX::~MessagePumpGlibX() {
|
| +}
|
| +
|
| +bool MessagePumpGlibX::RunOnce(GMainContext* context, bool block) {
|
| + GdkDisplay* gdisp = gdk_display_get_default();
|
| + Display* display = GDK_DISPLAY_XDISPLAY(gdisp);
|
| + if (XPending(display)) {
|
| + XEvent xev;
|
| + XPeekEvent(display, &xev);
|
| + if (capture_x_events_[xev.type]) {
|
| + XNextEvent(display, &xev);
|
| +
|
| + DLOG(INFO) << "nom noming event";
|
| +
|
| + // TODO(sad): Create a GdkEvent from |xev| and pass it on to
|
| + // EventDispatcherX. The ultimate goal is to create a views::Event from
|
| + // |xev| and send it to a rootview. When done, the preceding DLOG will be
|
| + // removed.
|
| + } else {
|
| + // TODO(sad): A couple of extra events can still sneak in during this
|
| + g_main_context_iteration(context, FALSE);
|
| + }
|
| + }
|
| +
|
| + bool retvalue;
|
| + if (gdksource_) {
|
| + // Replace the dispatch callback of the GDK event source temporarily so that
|
| + // it doesn't read events from X.
|
| + gboolean (*cb)(GSource*, GSourceFunc, void*) =
|
| + gdksource_->source_funcs->dispatch;
|
| + gdksource_->source_funcs->dispatch = PlaceholderDispatch;
|
| +
|
| + dispatching_event_ = true;
|
| + retvalue = g_main_context_iteration(context, block);
|
| + dispatching_event_ = false;
|
| +
|
| + gdksource_->source_funcs->dispatch = cb;
|
| + } else {
|
| + retvalue = g_main_context_iteration(context, block);
|
| + }
|
| +
|
| + return retvalue;
|
| +}
|
| +
|
| +void MessagePumpGlibX::InitializeEventsToCapture(void) {
|
| + // TODO(sad): Decide which events we want to capture and update the tables
|
| + // accordingly.
|
| + capture_x_events_[KeyPress] = true;
|
| + capture_gdk_events_[GDK_KEY_PRESS] = true;
|
| +
|
| + capture_x_events_[KeyRelease] = true;
|
| + capture_gdk_events_[GDK_KEY_RELEASE] = true;
|
| +}
|
| +
|
| +void MessagePumpGlibX::EventDispatcherX(GdkEvent* event, gpointer data) {
|
| + MessagePumpGlibX* pump_x = reinterpret_cast<MessagePumpGlibX*>(data);
|
| +
|
| + if (!pump_x->gdksource_) {
|
| + pump_x->gdksource_ = g_main_current_source();
|
| + } else if (!pump_x->IsDispatchingEvent()) {
|
| + if (event->type != GDK_NOTHING &&
|
| + pump_x->capture_gdk_events_[event->type]) {
|
| + // TODO(sad): An X event is caught by the GDK handler. Put it back in the
|
| + // X queue so that we catch it in the next iteration. When done, the
|
| + // following DLOG statement will be removed.
|
| + DLOG(INFO) << "GDK ruined it!!";
|
| + }
|
| + }
|
| +
|
| + pump_x->DispatchEvents(event);
|
| +}
|
| +
|
| +} // namespace base
|
|
|