blob: f4320db0bbf938d6c210670ce5db2a5caa06a9a7 [file] [log] [blame]
// Copyright 2013 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 "ui/events/test/events_test_utils_x11.h"
#include <stddef.h>
#include <X11/extensions/XI2.h>
#include <X11/keysym.h>
#include <X11/X.h>
#include <X11/Xlib.h>
#include "base/logging.h"
#include "base/macros.h"
#include "ui/events/devices/x11/touch_factory_x11.h"
#include "ui/events/event_constants.h"
#include "ui/events/event_utils.h"
#include "ui/events/keycodes/keyboard_code_conversion_x.h"
namespace {
// Converts ui::EventType to state for X*Events.
unsigned int XEventState(int flags) {
return
((flags & ui::EF_SHIFT_DOWN) ? ShiftMask : 0) |
((flags & ui::EF_CAPS_LOCK_ON) ? LockMask : 0) |
((flags & ui::EF_CONTROL_DOWN) ? ControlMask : 0) |
((flags & ui::EF_ALT_DOWN) ? Mod1Mask : 0) |
((flags & ui::EF_NUM_LOCK_ON) ? Mod2Mask : 0) |
((flags & ui::EF_MOD3_DOWN) ? Mod3Mask : 0) |
((flags & ui::EF_COMMAND_DOWN) ? Mod4Mask : 0) |
((flags & ui::EF_ALTGR_DOWN) ? Mod5Mask : 0) |
((flags & ui::EF_LEFT_MOUSE_BUTTON) ? Button1Mask: 0) |
((flags & ui::EF_MIDDLE_MOUSE_BUTTON) ? Button2Mask: 0) |
((flags & ui::EF_RIGHT_MOUSE_BUTTON) ? Button3Mask: 0);
}
// Converts EventType to XKeyEvent type.
int XKeyEventType(ui::EventType type) {
switch (type) {
case ui::ET_KEY_PRESSED:
return KeyPress;
case ui::ET_KEY_RELEASED:
return KeyRelease;
default:
return 0;
}
}
// Converts EventType to XI2 event type.
int XIKeyEventType(ui::EventType type) {
switch (type) {
case ui::ET_KEY_PRESSED:
return XI_KeyPress;
case ui::ET_KEY_RELEASED:
return XI_KeyRelease;
default:
return 0;
}
}
int XIButtonEventType(ui::EventType type) {
switch (type) {
case ui::ET_MOUSEWHEEL:
case ui::ET_MOUSE_PRESSED:
// The button release X events for mouse wheels are dropped by Aura.
return XI_ButtonPress;
case ui::ET_MOUSE_RELEASED:
return XI_ButtonRelease;
default:
NOTREACHED();
return 0;
}
}
// Converts Aura event type and flag to X button event.
unsigned int XButtonEventButton(ui::EventType type,
int flags) {
// Aura events don't keep track of mouse wheel button, so just return
// the first mouse wheel button.
if (type == ui::ET_MOUSEWHEEL)
return Button4;
if (flags & ui::EF_LEFT_MOUSE_BUTTON)
return Button1;
if (flags & ui::EF_MIDDLE_MOUSE_BUTTON)
return Button2;
if (flags & ui::EF_RIGHT_MOUSE_BUTTON)
return Button3;
return 0;
}
void InitValuatorsForXIDeviceEvent(XIDeviceEvent* xiev) {
int valuator_count = ui::DeviceDataManagerX11::DT_LAST_ENTRY;
xiev->valuators.mask_len = (valuator_count / 8) + 1;
xiev->valuators.mask = new unsigned char[xiev->valuators.mask_len];
memset(xiev->valuators.mask, 0, xiev->valuators.mask_len);
xiev->valuators.values = new double[valuator_count];
}
XEvent* CreateXInput2Event(int deviceid,
int evtype,
int tracking_id,
const gfx::Point& location) {
XEvent* event = new XEvent;
memset(event, 0, sizeof(*event));
event->type = GenericEvent;
event->xcookie.data = new XIDeviceEvent;
XIDeviceEvent* xiev =
static_cast<XIDeviceEvent*>(event->xcookie.data);
memset(xiev, 0, sizeof(XIDeviceEvent));
xiev->deviceid = deviceid;
xiev->sourceid = deviceid;
xiev->evtype = evtype;
xiev->detail = tracking_id;
xiev->event_x = location.x();
xiev->event_y = location.y();
xiev->event = DefaultRootWindow(gfx::GetXDisplay());
if (evtype == XI_ButtonPress || evtype == XI_ButtonRelease) {
xiev->buttons.mask_len = 8;
xiev->buttons.mask = new unsigned char[xiev->buttons.mask_len];
memset(xiev->buttons.mask, 0, xiev->buttons.mask_len);
}
return event;
}
} // namespace
namespace ui {
// XInput2 events contain additional data that need to be explicitly freed (see
// |CreateXInput2Event()|.
void XEventDeleter::operator()(XEvent* event) {
if (event->type == GenericEvent) {
XIDeviceEvent* xiev =
static_cast<XIDeviceEvent*>(event->xcookie.data);
if (xiev) {
delete[] xiev->valuators.mask;
delete[] xiev->valuators.values;
delete[] xiev->buttons.mask;
delete xiev;
}
}
delete event;
}
ScopedXI2Event::ScopedXI2Event() {}
ScopedXI2Event::~ScopedXI2Event() {}
void ScopedXI2Event::InitKeyEvent(EventType type,
KeyboardCode key_code,
int flags) {
XDisplay* display = gfx::GetXDisplay();
event_.reset(new XEvent);
memset(event_.get(), 0, sizeof(XEvent));
event_->type = XKeyEventType(type);
CHECK_NE(0, event_->type);
event_->xkey.serial = 0;
event_->xkey.send_event = 0;
event_->xkey.display = display;
event_->xkey.time = 0;
event_->xkey.window = 0;
event_->xkey.root = 0;
event_->xkey.subwindow = 0;
event_->xkey.x = 0;
event_->xkey.y = 0;
event_->xkey.x_root = 0;
event_->xkey.y_root = 0;
event_->xkey.state = XEventState(flags);
event_->xkey.keycode = XKeyCodeForWindowsKeyCode(key_code, flags, display);
event_->xkey.same_screen = 1;
}
void ScopedXI2Event::InitMotionEvent(const gfx::Point& location,
const gfx::Point& root_location,
int flags) {
XDisplay* display = gfx::GetXDisplay();
event_.reset(new XEvent);
memset(event_.get(), 0, sizeof(XEvent));
event_->type = MotionNotify;
event_->xmotion.serial = 0;
event_->xmotion.send_event = 0;
event_->xmotion.display = display;
event_->xmotion.time = 0;
event_->xmotion.window = 0;
event_->xmotion.root = 0;
event_->xkey.subwindow = 0;
event_->xmotion.x = location.x();
event_->xmotion.y = location.y();
event_->xmotion.x_root = root_location.x();
event_->xmotion.y_root = root_location.y();
event_->xmotion.state = XEventState(flags);
event_->xmotion.same_screen = 1;
}
void ScopedXI2Event::InitGenericKeyEvent(int deviceid,
int sourceid,
EventType type,
KeyboardCode key_code,
int flags) {
event_.reset(
CreateXInput2Event(deviceid, XIKeyEventType(type), 0, gfx::Point()));
XIDeviceEvent* xievent = static_cast<XIDeviceEvent*>(event_->xcookie.data);
CHECK_NE(0, xievent->evtype);
XDisplay* display = gfx::GetXDisplay();
event_->xgeneric.display = display;
xievent->display = display;
xievent->mods.effective = XEventState(flags);
xievent->detail = XKeyCodeForWindowsKeyCode(key_code, flags, display);
xievent->sourceid = sourceid;
}
void ScopedXI2Event::InitGenericButtonEvent(int deviceid,
EventType type,
const gfx::Point& location,
int flags) {
event_.reset(CreateXInput2Event(deviceid,
XIButtonEventType(type), 0, gfx::Point()));
XIDeviceEvent* xievent = static_cast<XIDeviceEvent*>(event_->xcookie.data);
xievent->mods.effective = XEventState(flags);
xievent->detail = XButtonEventButton(type, flags);
xievent->event_x = location.x();
xievent->event_y = location.y();
XISetMask(xievent->buttons.mask, xievent->detail);
// Setup an empty valuator list for generic button events.
SetUpValuators(std::vector<Valuator>());
}
void ScopedXI2Event::InitGenericMouseWheelEvent(int deviceid,
int wheel_delta,
int flags) {
InitGenericButtonEvent(deviceid, ui::ET_MOUSEWHEEL, gfx::Point(), flags);
XIDeviceEvent* xievent = static_cast<XIDeviceEvent*>(event_->xcookie.data);
xievent->detail = wheel_delta > 0 ? Button4 : Button5;
}
void ScopedXI2Event::InitScrollEvent(int deviceid,
int x_offset,
int y_offset,
int x_offset_ordinal,
int y_offset_ordinal,
int finger_count) {
event_.reset(CreateXInput2Event(deviceid, XI_Motion, 0, gfx::Point()));
Valuator valuators[] = {
Valuator(DeviceDataManagerX11::DT_CMT_SCROLL_X, x_offset),
Valuator(DeviceDataManagerX11::DT_CMT_SCROLL_Y, y_offset),
Valuator(DeviceDataManagerX11::DT_CMT_ORDINAL_X, x_offset_ordinal),
Valuator(DeviceDataManagerX11::DT_CMT_ORDINAL_Y, y_offset_ordinal),
Valuator(DeviceDataManagerX11::DT_CMT_FINGER_COUNT, finger_count)
};
SetUpValuators(
std::vector<Valuator>(valuators, valuators + arraysize(valuators)));
}
void ScopedXI2Event::InitFlingScrollEvent(int deviceid,
int x_velocity,
int y_velocity,
int x_velocity_ordinal,
int y_velocity_ordinal,
bool is_cancel) {
event_.reset(CreateXInput2Event(deviceid, XI_Motion, deviceid, gfx::Point()));
Valuator valuators[] = {
Valuator(DeviceDataManagerX11::DT_CMT_FLING_STATE, is_cancel ? 1 : 0),
Valuator(DeviceDataManagerX11::DT_CMT_FLING_Y, y_velocity),
Valuator(DeviceDataManagerX11::DT_CMT_ORDINAL_Y, y_velocity_ordinal),
Valuator(DeviceDataManagerX11::DT_CMT_FLING_X, x_velocity),
Valuator(DeviceDataManagerX11::DT_CMT_ORDINAL_X, x_velocity_ordinal)
};
SetUpValuators(
std::vector<Valuator>(valuators, valuators + arraysize(valuators)));
}
void ScopedXI2Event::InitTouchEvent(int deviceid,
int evtype,
int tracking_id,
const gfx::Point& location,
const std::vector<Valuator>& valuators) {
event_.reset(CreateXInput2Event(deviceid, evtype, tracking_id, location));
// If a timestamp was specified, setup the event.
for (size_t i = 0; i < valuators.size(); ++i) {
if (valuators[i].data_type ==
DeviceDataManagerX11::DT_TOUCH_RAW_TIMESTAMP) {
SetUpValuators(valuators);
return;
}
}
// No timestamp was specified. Use |ui::EventTimeForNow()|.
std::vector<Valuator> valuators_with_time = valuators;
valuators_with_time.push_back(
Valuator(DeviceDataManagerX11::DT_TOUCH_RAW_TIMESTAMP,
(ui::EventTimeForNow()).InMicroseconds()));
SetUpValuators(valuators_with_time);
}
void ScopedXI2Event::SetUpValuators(const std::vector<Valuator>& valuators) {
CHECK(event_.get());
CHECK_EQ(GenericEvent, event_->type);
XIDeviceEvent* xiev = static_cast<XIDeviceEvent*>(event_->xcookie.data);
InitValuatorsForXIDeviceEvent(xiev);
ui::DeviceDataManagerX11* manager = ui::DeviceDataManagerX11::GetInstance();
for (size_t i = 0; i < valuators.size(); ++i) {
manager->SetValuatorDataForTest(xiev, valuators[i].data_type,
valuators[i].value);
}
}
void SetUpTouchPadForTest(int deviceid) {
std::vector<int> device_list;
device_list.push_back(deviceid);
TouchFactory::GetInstance()->SetPointerDeviceForTest(device_list);
ui::DeviceDataManagerX11* manager = ui::DeviceDataManagerX11::GetInstance();
manager->SetDeviceListForTest(std::vector<int>(), device_list,
std::vector<int>());
}
void SetUpTouchDevicesForTest(const std::vector<int>& devices) {
TouchFactory::GetInstance()->SetTouchDeviceForTest(devices);
ui::DeviceDataManagerX11* manager = ui::DeviceDataManagerX11::GetInstance();
manager->SetDeviceListForTest(devices, std::vector<int>(),
std::vector<int>());
}
void SetUpPointerDevicesForTest(const std::vector<int>& devices) {
TouchFactory::GetInstance()->SetPointerDeviceForTest(devices);
ui::DeviceDataManagerX11* manager = ui::DeviceDataManagerX11::GetInstance();
manager->SetDeviceListForTest(std::vector<int>(), std::vector<int>(),
devices);
}
} // namespace ui