OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 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 "ui/events/x/hotplug_event_handler_x11.h" | 5 #include "ui/events/x/hotplug_event_handler_x11.h" |
6 | 6 |
7 #include <X11/extensions/XInput.h> | 7 #include <X11/extensions/XInput.h> |
8 #include <X11/extensions/XInput2.h> | 8 #include <X11/extensions/XInput2.h> |
9 | 9 |
| 10 #include <algorithm> |
10 #include <cmath> | 11 #include <cmath> |
11 #include <set> | 12 #include <set> |
12 #include <string> | 13 #include <string> |
13 #include <vector> | 14 #include <vector> |
14 | 15 |
15 #include "base/command_line.h" | 16 #include "base/command_line.h" |
16 #include "base/files/file_enumerator.h" | 17 #include "base/files/file_enumerator.h" |
17 #include "base/logging.h" | 18 #include "base/logging.h" |
18 #include "base/process/launch.h" | 19 #include "base/process/launch.h" |
19 #include "base/strings/string_util.h" | 20 #include "base/strings/string_util.h" |
20 #include "base/sys_info.h" | 21 #include "base/sys_info.h" |
21 #include "ui/events/device_hotplug_event_observer.h" | 22 #include "ui/events/device_hotplug_event_observer.h" |
| 23 #include "ui/events/input_device.h" |
| 24 #include "ui/events/keyboard_device.h" |
22 #include "ui/events/touchscreen_device.h" | 25 #include "ui/events/touchscreen_device.h" |
23 #include "ui/gfx/x/x11_types.h" | 26 #include "ui/gfx/x/x11_types.h" |
24 | 27 |
25 namespace ui { | 28 namespace ui { |
26 | 29 |
27 namespace { | 30 namespace { |
28 | 31 |
| 32 // The name of the xinput device corresponding to the internal keyboard. |
| 33 const char kInternalKeyboardName[] = "AT Translated Set 2 keyboard"; |
| 34 |
| 35 // Returns true if |name| is the name of a known keyboard device. Note, this may |
| 36 // return false negatives. |
| 37 bool IsKnownKeyboard(const std::string& name) { |
| 38 std::string lower(name); |
| 39 std::transform(lower.begin(), lower.end(), lower.begin(), ::tolower); |
| 40 return lower.find("keyboard") != std::string::npos; |
| 41 } |
| 42 |
| 43 // Returns true if |name| is the name of a known XTEST device. Note, this may |
| 44 // return false negatives. |
| 45 bool IsTestKeyboard(const std::string& name) { |
| 46 return name.find("XTEST") != std::string::npos; |
| 47 } |
| 48 |
29 // We consider the touchscreen to be internal if it is an I2c device. | 49 // We consider the touchscreen to be internal if it is an I2c device. |
30 // With the device id, we can query X to get the device's dev input | 50 // With the device id, we can query X to get the device's dev input |
31 // node eventXXX. Then we search all the dev input nodes registered | 51 // node eventXXX. Then we search all the dev input nodes registered |
32 // by I2C devices to see if we can find eventXXX. | 52 // by I2C devices to see if we can find eventXXX. |
33 bool IsTouchscreenInternal(XDisplay* dpy, int device_id) { | 53 bool IsTouchscreenInternal(XDisplay* dpy, int device_id) { |
34 using base::FileEnumerator; | 54 using base::FileEnumerator; |
35 using base::FilePath; | 55 using base::FilePath; |
36 | 56 |
37 #if !defined(CHROMEOS) | 57 #if !defined(CHROMEOS) |
38 return false; | 58 return false; |
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
109 : delegate_(delegate) { | 129 : delegate_(delegate) { |
110 } | 130 } |
111 | 131 |
112 HotplugEventHandlerX11::~HotplugEventHandlerX11() { | 132 HotplugEventHandlerX11::~HotplugEventHandlerX11() { |
113 } | 133 } |
114 | 134 |
115 void HotplugEventHandlerX11::OnHotplugEvent() { | 135 void HotplugEventHandlerX11::OnHotplugEvent() { |
116 const XIDeviceList& device_list = | 136 const XIDeviceList& device_list = |
117 DeviceListCacheX::GetInstance()->GetXI2DeviceList(gfx::GetXDisplay()); | 137 DeviceListCacheX::GetInstance()->GetXI2DeviceList(gfx::GetXDisplay()); |
118 HandleTouchscreenDevices(device_list); | 138 HandleTouchscreenDevices(device_list); |
| 139 HandleKeyboardDevices(device_list); |
| 140 } |
| 141 |
| 142 void HotplugEventHandlerX11::HandleKeyboardDevices( |
| 143 const XIDeviceList& x11_devices) { |
| 144 std::vector<KeyboardDevice> devices; |
| 145 |
| 146 for (int i = 0; i < x11_devices.count; i++) { |
| 147 if (!x11_devices[i].enabled || x11_devices[i].use != XISlaveKeyboard) |
| 148 continue; // Assume all keyboards are keyboard slaves |
| 149 std::string device_name(x11_devices[i].name); |
| 150 base::TrimWhitespaceASCII(device_name, base::TRIM_TRAILING, |
| 151 &device_name); |
| 152 InputDeviceType type; |
| 153 if (IsTestKeyboard(device_name)) |
| 154 continue; // Skip test devices. |
| 155 |
| 156 // TODO(rsadam@): Come up with a more generic way of identifying internal |
| 157 // keyboards. See crbug.com/420728. |
| 158 if (device_name == kInternalKeyboardName) { |
| 159 type = InputDeviceType::INTERNAL; |
| 160 } else if (IsKnownKeyboard(device_name)) { |
| 161 type = InputDeviceType::EXTERNAL; |
| 162 } else { |
| 163 type = InputDeviceType::UNKNOWN; |
| 164 } |
| 165 devices.push_back(KeyboardDevice(x11_devices[i].deviceid, |
| 166 type, device_name)); |
| 167 } |
| 168 delegate_->OnKeyboardDevicesUpdated(devices); |
119 } | 169 } |
120 | 170 |
121 void HotplugEventHandlerX11::HandleTouchscreenDevices( | 171 void HotplugEventHandlerX11::HandleTouchscreenDevices( |
122 const XIDeviceList& x11_devices) { | 172 const XIDeviceList& x11_devices) { |
123 std::vector<TouchscreenDevice> devices; | 173 std::vector<TouchscreenDevice> devices; |
124 Display* display = gfx::GetXDisplay(); | 174 Display* display = gfx::GetXDisplay(); |
125 Atom valuator_x = XInternAtom(display, "Abs MT Position X", False); | 175 Atom valuator_x = XInternAtom(display, "Abs MT Position X", False); |
126 Atom valuator_y = XInternAtom(display, "Abs MT Position Y", False); | 176 Atom valuator_y = XInternAtom(display, "Abs MT Position Y", False); |
127 if (valuator_x == None || valuator_y == None) | 177 if (valuator_x == None || valuator_y == None) |
128 return; | 178 return; |
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
162 XITouchClassInfo* touch_info = | 212 XITouchClassInfo* touch_info = |
163 reinterpret_cast<XITouchClassInfo*>(class_info); | 213 reinterpret_cast<XITouchClassInfo*>(class_info); |
164 is_direct_touch = touch_info->mode == XIDirectTouch; | 214 is_direct_touch = touch_info->mode == XIDirectTouch; |
165 } | 215 } |
166 #endif | 216 #endif |
167 } | 217 } |
168 | 218 |
169 // Touchscreens should have absolute X and Y axes, and be direct touch | 219 // Touchscreens should have absolute X and Y axes, and be direct touch |
170 // devices. | 220 // devices. |
171 if (width > 0.0 && height > 0.0 && is_direct_touch) { | 221 if (width > 0.0 && height > 0.0 && is_direct_touch) { |
172 bool is_internal = | 222 InputDeviceType type = |
173 IsTouchscreenInternal(display, x11_devices[i].deviceid); | 223 IsTouchscreenInternal(display, x11_devices[i].deviceid) ? |
| 224 InputDeviceType::INTERNAL : InputDeviceType::EXTERNAL; |
| 225 std::string name(x11_devices[i].name); |
174 devices.push_back(TouchscreenDevice( | 226 devices.push_back(TouchscreenDevice( |
175 x11_devices[i].deviceid, gfx::Size(width, height), is_internal)); | 227 x11_devices[i].deviceid, type, name, gfx::Size(width, height))); |
176 } | 228 } |
177 } | 229 } |
178 | 230 |
179 delegate_->OnTouchscreenDevicesUpdated(devices); | 231 delegate_->OnTouchscreenDevicesUpdated(devices); |
180 } | 232 } |
181 | 233 |
182 } // namespace ui | 234 } // namespace ui |
OLD | NEW |