| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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 "remoting/host/event_executor.h" | 5 #include "remoting/host/event_executor.h" |
| 6 | 6 |
| 7 #include <X11/Xlib.h> | 7 #include <X11/Xlib.h> |
| 8 #include <X11/extensions/XTest.h> | 8 #include <X11/extensions/XTest.h> |
| 9 #include <X11/extensions/XInput.h> | 9 #include <X11/extensions/XInput.h> |
| 10 | 10 |
| 11 #include <set> | 11 #include <set> |
| 12 | 12 |
| 13 #include "base/basictypes.h" | 13 #include "base/basictypes.h" |
| 14 #include "base/bind.h" | 14 #include "base/bind.h" |
| 15 #include "base/compiler_specific.h" | 15 #include "base/compiler_specific.h" |
| 16 #include "base/location.h" | 16 #include "base/location.h" |
| 17 #include "base/logging.h" | 17 #include "base/logging.h" |
| 18 #include "base/single_thread_task_runner.h" | 18 #include "base/single_thread_task_runner.h" |
| 19 #include "remoting/host/clipboard.h" |
| 19 #include "remoting/proto/internal.pb.h" | 20 #include "remoting/proto/internal.pb.h" |
| 20 #include "third_party/skia/include/core/SkPoint.h" | 21 #include "third_party/skia/include/core/SkPoint.h" |
| 21 | 22 |
| 22 namespace remoting { | 23 namespace remoting { |
| 23 | 24 |
| 24 namespace { | 25 namespace { |
| 25 | 26 |
| 26 using protocol::ClipboardEvent; | 27 using protocol::ClipboardEvent; |
| 27 using protocol::KeyEvent; | 28 using protocol::KeyEvent; |
| 28 using protocol::MouseEvent; | 29 using protocol::MouseEvent; |
| (...skipping 23 matching lines...) Expand all Loading... |
| 52 // EventExecutor interface. | 53 // EventExecutor interface. |
| 53 virtual void Start( | 54 virtual void Start( |
| 54 scoped_ptr<protocol::ClipboardStub> client_clipboard) OVERRIDE; | 55 scoped_ptr<protocol::ClipboardStub> client_clipboard) OVERRIDE; |
| 55 virtual void StopAndDelete() OVERRIDE; | 56 virtual void StopAndDelete() OVERRIDE; |
| 56 | 57 |
| 57 private: | 58 private: |
| 58 // Number of buttons we support. | 59 // Number of buttons we support. |
| 59 // Left, Right, Middle, VScroll Up/Down, HScroll Left/Right. | 60 // Left, Right, Middle, VScroll Up/Down, HScroll Left/Right. |
| 60 static const int kNumPointerButtons = 7; | 61 static const int kNumPointerButtons = 7; |
| 61 | 62 |
| 63 void InitClipboard(); |
| 64 |
| 62 // |mode| is one of the AutoRepeatModeOn, AutoRepeatModeOff, | 65 // |mode| is one of the AutoRepeatModeOn, AutoRepeatModeOff, |
| 63 // AutoRepeatModeDefault constants defined by the XChangeKeyboardControl() | 66 // AutoRepeatModeDefault constants defined by the XChangeKeyboardControl() |
| 64 // API. | 67 // API. |
| 65 void SetAutoRepeatForKey(int keycode, int mode); | 68 void SetAutoRepeatForKey(int keycode, int mode); |
| 66 void InjectScrollWheelClicks(int button, int count); | 69 void InjectScrollWheelClicks(int button, int count); |
| 67 // Compensates for global button mappings and resets the XTest device mapping. | 70 // Compensates for global button mappings and resets the XTest device mapping. |
| 68 void InitMouseButtonMap(); | 71 void InitMouseButtonMap(); |
| 69 int MouseButtonToX11ButtonNumber(MouseEvent::MouseButton button); | 72 int MouseButtonToX11ButtonNumber(MouseEvent::MouseButton button); |
| 70 int HorizontalScrollWheelToX11ButtonNumber(int dx); | 73 int HorizontalScrollWheelToX11ButtonNumber(int dx); |
| 71 int VerticalScrollWheelToX11ButtonNumber(int dy); | 74 int VerticalScrollWheelToX11ButtonNumber(int dy); |
| 72 | 75 |
| 73 scoped_refptr<base::SingleThreadTaskRunner> task_runner_; | 76 scoped_refptr<base::SingleThreadTaskRunner> task_runner_; |
| 74 | 77 |
| 75 std::set<int> pressed_keys_; | 78 std::set<int> pressed_keys_; |
| 76 SkIPoint latest_mouse_position_; | 79 SkIPoint latest_mouse_position_; |
| 77 | 80 |
| 78 // X11 graphics context. | 81 // X11 graphics context. |
| 79 Display* display_; | 82 Display* display_; |
| 80 Window root_window_; | 83 Window root_window_; |
| 81 | 84 |
| 82 int test_event_base_; | 85 int test_event_base_; |
| 83 int test_error_base_; | 86 int test_error_base_; |
| 84 | 87 |
| 85 int pointer_button_map_[kNumPointerButtons]; | 88 int pointer_button_map_[kNumPointerButtons]; |
| 89 |
| 90 scoped_ptr<Clipboard> clipboard_; |
| 91 |
| 86 DISALLOW_COPY_AND_ASSIGN(EventExecutorLinux); | 92 DISALLOW_COPY_AND_ASSIGN(EventExecutorLinux); |
| 87 }; | 93 }; |
| 88 | 94 |
| 89 EventExecutorLinux::EventExecutorLinux( | 95 EventExecutorLinux::EventExecutorLinux( |
| 90 scoped_refptr<base::SingleThreadTaskRunner> task_runner) | 96 scoped_refptr<base::SingleThreadTaskRunner> task_runner) |
| 91 : task_runner_(task_runner), | 97 : task_runner_(task_runner), |
| 92 latest_mouse_position_(SkIPoint::Make(-1, -1)), | 98 latest_mouse_position_(SkIPoint::Make(-1, -1)), |
| 93 display_(XOpenDisplay(NULL)), | 99 display_(XOpenDisplay(NULL)), |
| 94 root_window_(BadValue) { | 100 root_window_(BadValue) { |
| 101 if (!task_runner_->BelongsToCurrentThread()) { |
| 102 task_runner_->PostTask( |
| 103 FROM_HERE, |
| 104 base::Bind(&EventExecutorLinux::InitClipboard, base::Unretained(this))); |
| 105 } |
| 95 } | 106 } |
| 96 | 107 |
| 97 EventExecutorLinux::~EventExecutorLinux() { | 108 EventExecutorLinux::~EventExecutorLinux() { |
| 98 CHECK(pressed_keys_.empty()); | 109 CHECK(pressed_keys_.empty()); |
| 99 } | 110 } |
| 100 | 111 |
| 101 bool EventExecutorLinux::Init() { | 112 bool EventExecutorLinux::Init() { |
| 102 CHECK(display_); | 113 CHECK(display_); |
| 103 | 114 |
| 104 root_window_ = RootWindow(display_, DefaultScreen(display_)); | 115 root_window_ = RootWindow(display_, DefaultScreen(display_)); |
| 105 if (root_window_ == BadValue) { | 116 if (root_window_ == BadValue) { |
| 106 LOG(ERROR) << "Unable to get the root window"; | 117 LOG(ERROR) << "Unable to get the root window"; |
| 107 return false; | 118 return false; |
| 108 } | 119 } |
| 109 | 120 |
| 110 // TODO(ajwong): Do we want to check the major/minor version at all for XTest? | 121 // TODO(ajwong): Do we want to check the major/minor version at all for XTest? |
| 111 int major = 0; | 122 int major = 0; |
| 112 int minor = 0; | 123 int minor = 0; |
| 113 if (!XTestQueryExtension(display_, &test_event_base_, &test_error_base_, | 124 if (!XTestQueryExtension(display_, &test_event_base_, &test_error_base_, |
| 114 &major, &minor)) { | 125 &major, &minor)) { |
| 115 LOG(ERROR) << "Server does not support XTest."; | 126 LOG(ERROR) << "Server does not support XTest."; |
| 116 return false; | 127 return false; |
| 117 } | 128 } |
| 118 InitMouseButtonMap(); | 129 InitMouseButtonMap(); |
| 119 return true; | 130 return true; |
| 120 } | 131 } |
| 121 | 132 |
| 122 void EventExecutorLinux::InjectClipboardEvent(const ClipboardEvent& event) { | 133 void EventExecutorLinux::InjectClipboardEvent(const ClipboardEvent& event) { |
| 123 // TODO(simonmorris): Implement clipboard injection. | 134 if (!task_runner_->BelongsToCurrentThread()) { |
| 135 task_runner_->PostTask( |
| 136 FROM_HERE, |
| 137 base::Bind(&EventExecutorLinux::InjectClipboardEvent, |
| 138 base::Unretained(this), event)); |
| 139 return; |
| 140 } |
| 141 |
| 142 clipboard_->InjectClipboardEvent(event); |
| 124 } | 143 } |
| 125 | 144 |
| 126 void EventExecutorLinux::InjectKeyEvent(const KeyEvent& event) { | 145 void EventExecutorLinux::InjectKeyEvent(const KeyEvent& event) { |
| 127 // HostEventDispatcher should filter events missing the pressed field. | 146 // HostEventDispatcher should filter events missing the pressed field. |
| 128 DCHECK(event.has_pressed()); | 147 DCHECK(event.has_pressed()); |
| 129 DCHECK(event.has_usb_keycode()); | 148 DCHECK(event.has_usb_keycode()); |
| 130 | 149 |
| 131 if (!task_runner_->BelongsToCurrentThread()) { | 150 if (!task_runner_->BelongsToCurrentThread()) { |
| 132 task_runner_->PostTask( | 151 task_runner_->PostTask( |
| 133 FROM_HERE, | 152 FROM_HERE, |
| (...skipping 29 matching lines...) Expand all Loading... |
| 163 // case, this ensures that key-repeating will continue to work normally | 182 // case, this ensures that key-repeating will continue to work normally |
| 164 // for the local user of the host machine. "ModeDefault" is used instead | 183 // for the local user of the host machine. "ModeDefault" is used instead |
| 165 // of "ModeOn", since some keys (such as Shift) should not auto-repeat. | 184 // of "ModeOn", since some keys (such as Shift) should not auto-repeat. |
| 166 SetAutoRepeatForKey(keycode, AutoRepeatModeDefault); | 185 SetAutoRepeatForKey(keycode, AutoRepeatModeDefault); |
| 167 } | 186 } |
| 168 | 187 |
| 169 XTestFakeKeyEvent(display_, keycode, event.pressed(), CurrentTime); | 188 XTestFakeKeyEvent(display_, keycode, event.pressed(), CurrentTime); |
| 170 XFlush(display_); | 189 XFlush(display_); |
| 171 } | 190 } |
| 172 | 191 |
| 192 void EventExecutorLinux::InitClipboard() { |
| 193 DCHECK(task_runner_->BelongsToCurrentThread()); |
| 194 clipboard_ = Clipboard::Create(); |
| 195 } |
| 196 |
| 173 void EventExecutorLinux::SetAutoRepeatForKey(int keycode, int mode) { | 197 void EventExecutorLinux::SetAutoRepeatForKey(int keycode, int mode) { |
| 174 XKeyboardControl control; | 198 XKeyboardControl control; |
| 175 control.key = keycode; | 199 control.key = keycode; |
| 176 control.auto_repeat_mode = mode; | 200 control.auto_repeat_mode = mode; |
| 177 XChangeKeyboardControl(display_, KBKey | KBAutoRepeatMode, &control); | 201 XChangeKeyboardControl(display_, KBKey | KBAutoRepeatMode, &control); |
| 178 } | 202 } |
| 179 | 203 |
| 180 void EventExecutorLinux::InjectScrollWheelClicks(int button, int count) { | 204 void EventExecutorLinux::InjectScrollWheelClicks(int button, int count) { |
| 181 if (button < 0) { | 205 if (button < 0) { |
| 182 LOG(WARNING) << "Ignoring unmapped scroll wheel button"; | 206 LOG(WARNING) << "Ignoring unmapped scroll wheel button"; |
| (...skipping 171 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 354 void EventExecutorLinux::Start( | 378 void EventExecutorLinux::Start( |
| 355 scoped_ptr<protocol::ClipboardStub> client_clipboard) { | 379 scoped_ptr<protocol::ClipboardStub> client_clipboard) { |
| 356 if (!task_runner_->BelongsToCurrentThread()) { | 380 if (!task_runner_->BelongsToCurrentThread()) { |
| 357 task_runner_->PostTask( | 381 task_runner_->PostTask( |
| 358 FROM_HERE, | 382 FROM_HERE, |
| 359 base::Bind(&EventExecutorLinux::Start, | 383 base::Bind(&EventExecutorLinux::Start, |
| 360 base::Unretained(this), | 384 base::Unretained(this), |
| 361 base::Passed(&client_clipboard))); | 385 base::Passed(&client_clipboard))); |
| 362 return; | 386 return; |
| 363 } | 387 } |
| 388 |
| 364 InitMouseButtonMap(); | 389 InitMouseButtonMap(); |
| 365 return; | 390 clipboard_->Start(client_clipboard.Pass()); |
| 366 } | 391 } |
| 367 | 392 |
| 368 void EventExecutorLinux::StopAndDelete() { | 393 void EventExecutorLinux::StopAndDelete() { |
| 369 delete this; | 394 delete this; |
| 370 } | 395 } |
| 371 | 396 |
| 372 } // namespace | 397 } // namespace |
| 373 | 398 |
| 374 scoped_ptr<EventExecutor> EventExecutor::Create( | 399 scoped_ptr<EventExecutor> EventExecutor::Create( |
| 375 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, | 400 scoped_refptr<base::SingleThreadTaskRunner> main_task_runner, |
| 376 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner) { | 401 scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner) { |
| 377 scoped_ptr<EventExecutorLinux> executor( | 402 scoped_ptr<EventExecutorLinux> executor( |
| 378 new EventExecutorLinux(main_task_runner)); | 403 new EventExecutorLinux(main_task_runner)); |
| 379 if (!executor->Init()) | 404 if (!executor->Init()) |
| 380 return scoped_ptr<EventExecutor>(NULL); | 405 return scoped_ptr<EventExecutor>(NULL); |
| 381 return executor.PassAs<EventExecutor>(); | 406 return executor.PassAs<EventExecutor>(); |
| 382 } | 407 } |
| 383 | 408 |
| 384 } // namespace remoting | 409 } // namespace remoting |
| OLD | NEW |