Files
ukui-settings-daemon/plugins/mouse/mouse-manager.cpp
2020-11-25 14:31:57 +08:00

1727 lines
57 KiB
C++
Executable File

/* -*- Mode: C++; indent-tabs-mode: nil; tab-width: 4 -*-
* -*- coding: utf-8 -*-
*
* Copyright (C) 2020 KylinSoft Co., Ltd.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "mouse-manager.h"
#include "clib-syslog.h"
/* Keys with same names for both touchpad and mouse */
#define KEY_LEFT_HANDED "left-handed" /* a boolean for mouse, an enum for touchpad */
#define KEY_MOTION_ACCELERATION "motion-acceleration"
#define KEY_MOTION_THRESHOLD "motion-threshold"
/* Mouse settings */
#define UKUI_MOUSE_SCHEMA "org.ukui.peripherals-mouse"
#define KEY_MOUSE_LOCATE_POINTER "locate-pointer"
#define KEY_MIDDLE_BUTTON_EMULATION "middle-button-enabled"
#define KEY_MOUSE_WHEEL_SPEED "wheel-speed"
#define KEY_MOUSE_ACCEL "mouse-accel"
/* Touchpad settings */
#define UKUI_TOUCHPAD_SCHEMA "org.ukui.peripherals-touchpad"
#define KEY_TOUCHPAD_DISABLE_W_TYPING "disable-while-typing"
#define KEY_TOUCHPAD_TWO_FINGER_CLICK "two-finger-click"
#define KEY_TOUCHPAD_THREE_FINGER_CLICK "three-finger-click"
#define KEY_TOUCHPAD_NATURAL_SCROLL "natural-scroll"
#define KEY_TOUCHPAD_TAP_TO_CLICK "tap-to-click"
#define KEY_TOUCHPAD_ONE_FINGER_TAP "tap-button-one-finger"
#define KEY_TOUCHPAD_TWO_FINGER_TAP "tap-button-two-finger"
#define KEY_TOUCHPAD_THREE_FINGER_TAP "tap-button-three-finger"
#define KEY_VERT_EDGE_SCROLL "vertical-edge-scrolling"
#define KEY_HORIZ_EDGE_SCROLL "horizontal-edge-scrolling"
#define KEY_VERT_TWO_FINGER_SCROLL "vertical-two-finger-scrolling"
#define KEY_HORIZ_TWO_FINGER_SCROLL "horizontal-two-finger-scrolling"
#define KEY_TOUCHPAD_ENABLED "touchpad-enabled"
#define KEY_TOUCHPAD_DISBLE_O_E_MOUSE "disable-on-external-mouse" //插入鼠标,禁用触摸板 true/false
#define KEY_TOUCHPAD_DOUBLE_CLICK_DRAG "double-click-drag" //点击两次拖动 true/false
#define KEY_TOUCHPAD_BOTTOM_R_C_CLICK_M "bottom-right-corner-click-menu" //右下角点击菜单 true/false
#define KEY_TOUCHPAD_MOUSE_SENSITVITY "mouse-sensitivity" //鼠标敏感度 1-4 四个档位 低中高最高
typedef enum {
TOUCHPAD_HANDEDNESS_RIGHT,
TOUCHPAD_HANDEDNESS_LEFT,
TOUCHPAD_HANDEDNESS_MOUSE
} TouchpadHandedness;
GdkFilterReturn devicepresence_filter (GdkXEvent *xevent,
GdkEvent *event,
gpointer data);
bool supports_xinput_devices (void);
bool touchpad_is_present (void);
MouseManager * MouseManager::mMouseManager =nullptr;
MouseManager::MouseManager(QObject *parent) : QObject (parent)
{
gdk_init(NULL,NULL);
syndaemon_spawned = false;
syndaemon_pid = 0;
locate_pointer_spawned = false;
locate_pointer_pid = 0;
imwheelSpawned = false;
settings_mouse = new QGSettings(UKUI_MOUSE_SCHEMA);
settings_touchpad = new QGSettings(UKUI_TOUCHPAD_SCHEMA);
}
MouseManager::~MouseManager()
{
delete settings_mouse;
delete settings_touchpad;
if(time)
delete time;
}
MouseManager * MouseManager::MouseManagerNew()
{
if(nullptr == mMouseManager)
mMouseManager = new MouseManager(nullptr);
return mMouseManager;
}
bool MouseManager::MouseManagerStart()
{
CT_SYSLOG(LOG_DEBUG,"-- Mouse Start Manager --");
if (!supports_xinput_devices()){
qWarning("XInput is not supported, not applying any settings");
return TRUE;
}
time = new QTimer(this);
connect(time,SIGNAL(timeout()),this,SLOT(MouseManagerIdleCb()));
time->start();
return true;
}
void MouseManager::MouseManagerStop()
{
syslog(LOG_DEBUG,"-- Stopping Mouse Manager --");
SetLocatePointer(FALSE);
gdk_window_remove_filter (NULL, devicepresence_filter, this);
}
/* transplant usd-input-helper.h */
/* Checks whether the XInput device is supported
* 检测是否支持xinput设备
*/
bool
supports_xinput_devices (void)
{
int op_code, event, error;
return XQueryExtension (QX11Info::display(),
"XInputExtension",
&op_code,
&event,
&error);
}
static bool
device_has_property (XDevice *device,
const char *property_name)
{
Atom realtype, prop;
int realformat;
unsigned long nitems, bytes_after;
unsigned char *data;
prop = XInternAtom (QX11Info::display(), property_name, True);
if (!prop)
return FALSE;
try {
if ((XGetDeviceProperty (QX11Info::display(), device, prop, 0, 1, False,
XA_INTEGER, &realtype, &realformat, &nitems,
&bytes_after, &data) == Success) && (realtype != None)) {
XFree (data);
return TRUE;
}
} catch (int x) {
}
return FALSE;
}
XDevice* device_is_touchpad (XDeviceInfo *deviceinfo)
{
XDevice *device;
if (deviceinfo->type != XInternAtom (QX11Info::display(), XI_TOUCHPAD, True))
return NULL;
try {
device = XOpenDevice (QX11Info::display(), deviceinfo->id);
if (device == NULL)
throw 1;
if (device_has_property (device, "libinput Tapping Enabled") ||
device_has_property (device, "Synaptics Off")) {
return device;
}
XCloseDevice (QX11Info::display(), device);
} catch (int x) {
return NULL;
}
return NULL;
}
bool touchpad_is_present (void)
{
XDeviceInfo *device_info;
int n_devices;
int i;
bool retval;
if (supports_xinput_devices () == false)
return true;
retval = false;
device_info = XListInputDevices (QX11Info::display(), &n_devices);
if (device_info == NULL)
return false;
for (i = 0; i < n_devices; i++) {
XDevice *device;
device = device_is_touchpad (&device_info[i]);
if (device != NULL) {
retval = true;
break;
}
}
if (device_info != NULL)
XFreeDeviceList (device_info);
return retval;
}
bool MouseManager::GetTouchpadHandedness (bool mouse_left_handed)
{
int a = settings_touchpad->getEnum(KEY_LEFT_HANDED);
switch (a) {
case TOUCHPAD_HANDEDNESS_RIGHT:
return false;
case TOUCHPAD_HANDEDNESS_LEFT:
return true;
case TOUCHPAD_HANDEDNESS_MOUSE:
return mouse_left_handed;
default:
g_assert_not_reached ();
}
}
Atom property_from_name (const char *property_name)
{
return XInternAtom (QX11Info::display(), property_name, True);
}
bool property_exists_on_device (XDeviceInfo *device_info, const char *property_name)
{
XDevice *device;
int rc;
Atom type, prop;
int format;
unsigned long nitems, bytes_after;
unsigned char *data;
Display *display = gdk_x11_get_default_xdisplay ();//QX11Info::display();
prop = property_from_name (property_name);
if (!prop)
return FALSE;
try {
device = XOpenDevice (display, device_info->id);
if (device == NULL)
throw 1;
rc = XGetDeviceProperty (display,
device, prop, 0, 1, False, XA_INTEGER, &type, &format,
&nitems, &bytes_after, &data);
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int str) {
CT_SYSLOG(LOG_DEBUG,"MOUSE: WRING ID: %d",str);
return FALSE;
}
return rc == Success;
}
void property_set_bool (XDeviceInfo *device_info,
XDevice *device,
const char *property_name,
int property_index,
bool enabled)
{
int rc;
unsigned long nitems, bytes_after;
unsigned char *data;
int act_format;
Atom act_type, property;
Display *display = gdk_x11_get_default_xdisplay ();//QX11Info::display();
property = property_from_name (property_name);
if (!property)
return;
try {
gdk_x11_display_error_trap_push (gdk_display_get_default());
rc = XGetDeviceProperty (display, device,
property, 0, 1, False,
XA_INTEGER, &act_type, &act_format, &nitems,
&bytes_after, &data);
if (rc == Success && act_type == XA_INTEGER && act_format == 8 && nitems >(unsigned long)property_index) {
data[property_index] = enabled ? 1 : 0;
XChangeDeviceProperty (display, device,
property, XA_INTEGER, 8,
PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
if(gdk_x11_display_error_trap_pop (gdk_display_get_default()))
qWarning ("Error while setting %s on \"%s\"", property_name, device_info->name);
} catch (int x) {
CT_SYSLOG(LOG_DEBUG,"MOUSE:Error while setting %s on \"%s\"", property_name, device_info->name)
}
}
void set_left_handed_libinput (XDeviceInfo *device_info,
bool mouse_left_handed,
bool touchpad_left_handed)
{
XDevice *device;
bool want_lefthanded;
Display *display = QX11Info::display();
device = device_is_touchpad (device_info);
try {
if (device == NULL) {
device = XOpenDevice (display, device_info->id);
if (device == NULL)
throw 1;
want_lefthanded = mouse_left_handed;
} else {
/* touchpad device is already open after
* return from device_is_touchpad function
*/
want_lefthanded = touchpad_left_handed;
}
property_set_bool (device_info, device, "libinput Left Handed Enabled", 0, want_lefthanded);
XCloseDevice (display, device);
} catch (int x) {
return;
}
}
gboolean xinput_device_has_buttons (XDeviceInfo *device_info)
{
int i;
XAnyClassInfo *class_info;
class_info = device_info->inputclassinfo;
for (i = 0; i < device_info->num_classes; i++) {
if (class_info->c_class == ButtonClass) {
XButtonInfo *button_info;
button_info = (XButtonInfo *) class_info;
if (button_info->num_buttons > 0)
return TRUE;
}
class_info = (XAnyClassInfo *) (((guchar *) class_info) +
class_info->length);
}
return FALSE;
}
bool touchpad_has_single_button (XDevice *device)
{
Atom type, prop;
int format;
unsigned long nitems, bytes_after;
unsigned char *data;
bool is_single_button = FALSE;
int rc;
prop = property_from_name ("Synaptics Capabilities");
if (!prop)
return false;
try {
rc = XGetDeviceProperty (QX11Info::display(), device, prop, 0, 1, False,
XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 8 && nitems >= 3)
is_single_button = (data[0] == 1 && data[1] == 0 && data[2] == 0);
if (rc == Success)
XFree (data);
} catch (int x) {
}
return is_single_button;
}
void set_tap_to_click_synaptics (XDeviceInfo *device_info,
bool state,
bool left_handed,
int one_finger_tap,
int two_finger_tap,
int three_finger_tap)
{
XDevice *device;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char* data;
Atom prop, type;
Display *display = gdk_x11_get_default_xdisplay (); //QX11Info::display();
prop = property_from_name ("Synaptics Tap Action");
if (!prop)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
try {
rc = XGetDeviceProperty (display, device, prop, 0, 2,
False, XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (one_finger_tap > 3 || one_finger_tap < 1)
one_finger_tap = 1;
if (two_finger_tap > 3 || two_finger_tap < 1)
two_finger_tap = 3;
if (three_finger_tap > 3 || three_finger_tap < 1)
three_finger_tap = 2;
if (rc == Success && type == XA_INTEGER && format == 8 && nitems >= 7)
{
/* Set RLM mapping for 1/2/3 fingers*/
data[4] = (state) ? ((left_handed) ? (4-one_finger_tap) : one_finger_tap) : 0;
data[5] = (state) ? ((left_handed) ? (4-two_finger_tap) : two_finger_tap) : 0;
data[6] = (state) ? three_finger_tap : 0;
XChangeDeviceProperty (display, device, prop, XA_INTEGER, 8,
PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting tap to click on \"%s\"", device_info->name);
}
}
void configure_button_layout (guchar *buttons,
int n_buttons,
bool left_handed)
{
const int left_button = 1;
int right_button;
int i;
/* if the button is higher than 2 (3rd button) then it's
* probably one direction of a scroll wheel or something else
* uninteresting
*/
right_button = MIN (n_buttons, 3);
/* If we change things we need to make sure we only swap buttons.
* If we end up with multiple physical buttons assigned to the same
* logical button the server will complain. This code assumes physical
* button 0 is the physical left mouse button, and that the physical
* button other than 0 currently assigned left_button or right_button
* is the physical right mouse button.
*/
/* check if the current mapping satisfies the above assumptions */
if (buttons[left_button - 1] != left_button &&
buttons[left_button - 1] != right_button)
/* The current mapping is weird. Swapping buttons is probably not a
* good idea.
*/
return;
/* check if we are left_handed and currently not swapped */
if (left_handed && buttons[left_button - 1] == left_button) {
/* find the right button */
for (i = 0; i < n_buttons; i++) {
if (buttons[i] == right_button) {
buttons[i] = left_button;
break;
}
}
/* swap the buttons */
buttons[left_button - 1] = right_button;
}
/* check if we are not left_handed but are swapped */
else if (!left_handed && buttons[left_button - 1] == right_button) {
/* find the right button */
for (i = 0; i < n_buttons; i++) {
if (buttons[i] == left_button) {
buttons[i] = right_button;
break;
}
}
/* swap the buttons */
buttons[left_button - 1] = left_button;
}
}
void MouseManager::SetLeftHandedLegacyDriver (XDeviceInfo *device_info,
bool mouse_left_handed,
bool touchpad_left_handed)
{
XDevice *device;
unsigned char *buttons;
unsigned long buttons_capacity = 16;
int n_buttons;
bool left_handed;
Display *display = QX11Info::display();
if ((device_info->use == IsXPointer) ||
(device_info->use == IsXKeyboard) ||
(g_strcmp0 ("Virtual core XTEST pointer", device_info->name) == 0) ||
(!xinput_device_has_buttons (device_info)))
return;
/* If the device is a touchpad, swap tap buttons
* around too, otherwise a tap would be a right-click */
device = device_is_touchpad (device_info);
if (device != NULL) {
bool tap = settings_touchpad->get(KEY_TOUCHPAD_TAP_TO_CLICK).toBool();
bool single_button = touchpad_has_single_button (device);
left_handed = touchpad_left_handed;
if (tap && !single_button) {
int one_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_ONE_FINGER_TAP).toInt();
int two_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_TWO_FINGER_TAP).toInt();
int three_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_THREE_FINGER_TAP).toInt();
set_tap_to_click_synaptics (device_info, tap, left_handed, one_finger_tap, two_finger_tap, three_finger_tap);
}
XCloseDevice (display, device);
if (single_button)
return;
} else {
left_handed = mouse_left_handed;
}
try {
device = XOpenDevice (display, device_info->id);
if (device == NULL)
throw 1;
buttons = g_new (guchar, buttons_capacity);
n_buttons = XGetDeviceButtonMapping (display, device,
buttons,
buttons_capacity);
while (n_buttons > (int)buttons_capacity) {
buttons_capacity = n_buttons;
buttons = (guchar *) g_realloc (buttons,
buttons_capacity * sizeof (guchar));
n_buttons = XGetDeviceButtonMapping (display, device,
buttons,
buttons_capacity);
}
configure_button_layout (buttons, n_buttons, left_handed);
XSetDeviceButtonMapping (display, device, buttons, n_buttons);
XCloseDevice (display, device);
g_free (buttons);
} catch (int x) {
CT_SYSLOG(LOG_DEBUG,"MOUSE :error id %d",x);
return;
}
}
void MouseManager::SetLeftHanded (XDeviceInfo *device_info,
bool mouse_left_handed,
bool touchpad_left_handed)
{
if (property_exists_on_device (device_info, "libinput Left Handed Enabled"))
set_left_handed_libinput (device_info, mouse_left_handed, touchpad_left_handed);
else
SetLeftHandedLegacyDriver (device_info, mouse_left_handed, touchpad_left_handed);
}
void MouseManager::SetLeftHandedAll (bool mouse_left_handed,
bool touchpad_left_handed)
{
XDeviceInfo *device_info = nullptr;
int n_devices;
int i;
Display * dpy = QX11Info::display();
device_info = XListInputDevices (dpy, &n_devices);
if(!device_info){
qWarning("SetLeftHandedAll: device_info is null");
return;
}
for (i = 0; i < n_devices; i++) {
SetLeftHanded (&device_info[i], mouse_left_handed, touchpad_left_handed);
}
if (device_info != NULL)
XFreeDeviceList (device_info);
}
void MouseManager::SetMotionLibinput (XDeviceInfo *device_info)
{
XDevice *device;
Atom prop;
Atom type;
Atom float_type;
int format, rc;
unsigned long nitems, bytes_after;
QGSettings *settings;
Display * dpy = GDK_DISPLAY_XDISPLAY (gdk_display_get_default ());// gdk_x11_get_default_xdisplay ();//QX11Info::display();
union {
unsigned char *c;
long *l;
} data;
float accel;
float motion_acceleration;
float_type = property_from_name ("FLOAT");
if (!float_type)
return;
prop = property_from_name ("libinput Accel Speed");
if (!prop) {
return;
}
try {
device = device_is_touchpad (device_info);
if (device != NULL) {
qDebug()<<"device != NULL settings = settings_touchpad";
settings = settings_touchpad;
} else {
device = XOpenDevice (dpy, device_info->id);
if (device == NULL)
throw 1;
settings = settings_mouse;
}
/* Calculate acceleration */
motion_acceleration = settings->get(KEY_MOTION_ACCELERATION).toDouble();
/* panel gives us a range of 1.0-10.0, map to libinput's [-1, 1]
*
* oldrange = (oldmax - oldmin)
* newrange = (newmax - newmin)
*
* mapped = (value - oldmin) * newrange / oldrange + oldmin
*/
if (motion_acceleration == -1.0) /* unset */
accel = 0.0;
else
accel = (motion_acceleration - 1.0) * 2.0 / 9.0 - 1;
rc = XGetDeviceProperty (dpy,device, prop, 0, 1, False, float_type, &type,
&format, &nitems, &bytes_after, &data.c);
if (rc == Success && type == float_type && format == 32 && nitems >= 1) {
*(float *) data.l = accel;
XChangeDeviceProperty (dpy, device, prop, float_type, 32,
PropModeReplace, data.c, nitems);
}
if (rc == Success) {
XFree (data.c);
}
XCloseDevice (dpy, device);
} catch (int x) {
qWarning("%s Error while setting accel speed on \"%s\"", device_info->name);
return;
}
}
void MouseManager::SetMotionLegacyDriver (XDeviceInfo *device_info)
{
XDevice *device;
XPtrFeedbackControl feedback;
XFeedbackState *states, *state;
int num_feedbacks, i;
QGSettings *settings;
double motion_acceleration;
int motion_threshold;
int numerator, denominator;
Display * dpy = gdk_x11_get_default_xdisplay ();//QX11Info::display();
device = device_is_touchpad (device_info);
if (device != NULL) {
settings = settings_touchpad;
} else {
try {
device = XOpenDevice (dpy, device_info->id);
if (device == NULL)
throw 1;
} catch (int x) {
CT_SYSLOG(LOG_DEBUG,"%s: error id %d","MOUSE", x);
return;
}
settings = settings_mouse;
}
/* Calculate acceleration */
motion_acceleration = settings->get(KEY_MOTION_ACCELERATION).toDouble();
if (motion_acceleration >= 1.0) {
/* we want to get the acceleration, with a resolution of 0.5
*/
if ((motion_acceleration - floor (motion_acceleration)) < 0.25) {
numerator = floor (motion_acceleration);
denominator = 1;
} else if ((motion_acceleration - floor (motion_acceleration)) < 0.5) {
numerator = ceil (2.0 * motion_acceleration);
denominator = 2;
} else if ((motion_acceleration - floor (motion_acceleration)) < 0.75) {
numerator = floor (2.0 *motion_acceleration);
denominator = 2;
} else {
numerator = ceil (motion_acceleration);
denominator = 1;
}
} else if (motion_acceleration < 1.0 && motion_acceleration > 0) {
/* This we do to 1/10ths */
numerator = floor (motion_acceleration * 10) + 1;
denominator= 10;
} else {
numerator = -1;
denominator = -1;
}
/* And threshold */
motion_threshold = settings->get(KEY_MOTION_THRESHOLD).toInt();
qDebug()<<__func__<<" motion_threshold = "<<motion_threshold;
/* Get the list of feedbacks for the device */
states = XGetFeedbackControl (dpy, device, &num_feedbacks);
if (states == NULL) {
XCloseDevice (dpy, device);
return;
}
state = (XFeedbackState *) states;
for (i = 0; i < num_feedbacks; i++) {
if (state->c_class == PtrFeedbackClass) {
/* And tell the device */
feedback.c_class = PtrFeedbackClass;
feedback.length = sizeof (XPtrFeedbackControl);
feedback.id = state->id;
feedback.threshold = motion_threshold;
feedback.accelNum = numerator;
feedback.accelDenom = denominator;
qDebug ("Setting accel %d/%d, threshold %d for device '%s'",
numerator, denominator, motion_threshold, device_info->name);
XChangeFeedbackControl (dpy,
device,
DvAccelNum | DvAccelDenom | DvThreshold,
(XFeedbackControl *) &feedback);
break;
}
state = (XFeedbackState *) ((char *) state + state->length);
}
XFreeFeedbackList (states);
XCloseDevice (dpy, device);
}
void MouseManager::SetTouchpadMotionAccel(XDeviceInfo *device_info)
{
XDevice *device;
Atom prop;
Atom type;
Atom float_type;
int format, rc;
unsigned long nitems, bytes_after;
Display * dpy = gdk_x11_get_default_xdisplay ();//QX11Info::display();
union {
unsigned char *c;
long *l;
} data;
float accel;
float motion_acceleration;
float_type = property_from_name ("FLOAT");
if (!float_type)
return;
prop = property_from_name ("Device Accel Constant Deceleration");
if (!prop) {
return;
}
try {
device = device_is_touchpad (device_info);
if (device == NULL)
return;
/* Calculate acceleration */
motion_acceleration = settings_touchpad->get(KEY_MOTION_ACCELERATION).toDouble();
if (motion_acceleration == -1.0) /* unset */
accel = 0.0;
else
accel = motion_acceleration;
rc = XGetDeviceProperty (dpy,device, prop, 0, 1, False, float_type, &type,
&format, &nitems, &bytes_after, &data.c);
qDebug()<<"format = "<<format<<"nitems = "<<nitems;
if (rc == Success && type == float_type && format == 32 && nitems >= 1) {
*(float *) data.l = accel;
XChangeDeviceProperty (dpy, device, prop, float_type, 32,
PropModeReplace, data.c, nitems);
}
if (rc == Success) {
XFree (data.c);
}
XCloseDevice (dpy, device);
} catch (int x) {
qWarning("%s Error while setting accel speed on \"%s\"", device_info->name);
return;
}
}
void MouseManager::SetMouseAccel(XDeviceInfo *device_info)
{
XDevice *device;
Atom prop;
Atom type;
int format, rc;
unsigned long nitems, bytes_after;
Display * dpy = QX11Info::display();
unsigned char *data;
bool MouseAccel;
prop = property_from_name ("libinput Accel Profile Enabled");
if (!prop) {
return;
}
try {
device = XOpenDevice (dpy, device_info->id);
if (device == NULL)
throw 1;
rc = XGetDeviceProperty (dpy,device, prop, 0, 2, False, XA_INTEGER, &type,
&format, &nitems, &bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 8 && nitems >= 1) {
MouseAccel = settings_mouse->get(KEY_MOUSE_ACCEL).toBool();
if(MouseAccel){
data[0] = 1;
data[1] = 0;
}else{
data[0] = 0;
data[1] = 1;
}
XChangeDeviceProperty (dpy, device, prop, XA_INTEGER, 8,
PropModeReplace, data, nitems);
}
if (rc == Success) {
XFree (data);
}
XCloseDevice (dpy, device);
} catch (int x) {
qWarning("%s Error while setting accel speed on \"%s\"", device_info->name);
return;
}
}
void MouseManager::SetMotion (XDeviceInfo *device_info)
{
if (property_exists_on_device (device_info, "libinput Accel Speed"))
SetMotionLibinput (device_info);
else
SetMotionLegacyDriver (device_info);
if(property_exists_on_device (device_info, "Device Accel Constant Deceleration"))
SetTouchpadMotionAccel(device_info);
if(property_exists_on_device (device_info, "libinput Accel Profile Enabled")) {
SetMouseAccel(device_info);
}
}
void MouseManager::SetMotionAll ()
{
XDeviceInfo *device_info = nullptr;
int n_devices;
int i;
device_info = XListInputDevices (gdk_x11_get_default_xdisplay (), &n_devices);
if(!device_info){
qWarning("SetMotionAll: device_info is null");
return;
}
for (i = 0; i < n_devices; i++) {
SetMotion (&device_info[i]);
}
if (device_info != NULL)
XFreeDeviceList (device_info);
}
void set_middle_button_evdev (XDeviceInfo *device_info,
bool middle_button)
{
XDevice *device;
Atom prop;
Atom type;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char *data;
Display * display = QX11Info::display();
prop = property_from_name ("Evdev Middle Button Emulation");
if (!prop) /* no evdev devices */
return;
try {
device = XOpenDevice (display, device_info->id);
if (device == NULL)
throw 1;
rc = XGetDeviceProperty (display,
device, prop, 0, 1, False, XA_INTEGER, &type, &format,
&nitems, &bytes_after, &data);
if (rc == Success && format == 8 && type == XA_INTEGER && nitems == 1) {
data[0] = middle_button ? 1 : 0;
XChangeDeviceProperty (display, device, prop, type, format, PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting middle button emulation on \"%s\"", device_info->name);
return;
}
}
void set_middle_button_libinput (XDeviceInfo *device_info,
bool middle_button)
{
XDevice *device;
Display *display = QX11Info::display();
/* touchpad devices are excluded as the old code
* only applies to evdev devices
*/
device = device_is_touchpad (device_info);
if (device != NULL) {
try {
XCloseDevice (display, device);
} catch (int x) {
return;
}
}
try {
device = XOpenDevice (display, device_info->id);
if (device == NULL)
throw 1;
property_set_bool (device_info, device, "libinput Middle Emulation Enabled", 0, middle_button);
XCloseDevice (display, device);
} catch (int x) {
CT_SYSLOG(LOG_DEBUG,"%s:error id %d","MOUSE",x);
return;
}
}
void MouseManager::SetMiddleButton (XDeviceInfo *device_info,
bool middle_button)
{
if (property_from_name ("Evdev Middle Button Emulation"))
set_middle_button_evdev (device_info, middle_button);
if (property_from_name ("libinput Middle Emulation Enabled"))
set_middle_button_libinput (device_info, middle_button);
}
void MouseManager::SetMiddleButtonAll (bool middle_button)
{
XDeviceInfo *device_info = nullptr;
int n_devices;
int i;
Display * display = QX11Info::display();
device_info = XListInputDevices (display, &n_devices);
if(!device_info){
qWarning("SetMiddleButtonAll: device_info is null");
return;
}
for (i = 0; i < n_devices; i++) {
SetMiddleButton (&device_info[i], middle_button);
}
if (device_info != NULL)
XFreeDeviceList (device_info);
}
void MouseManager::SetLocatePointer (bool state)
{
if (state) {
GError *error = NULL;
char **args;
int argc;
if (locate_pointer_spawned)
return;
QString str = "/usr/bin/usd-locate-pointer";
if( g_shell_parse_argv (str.toLatin1().data(), &argc, &args, NULL)){
g_spawn_async (g_get_home_dir (),
args,
NULL,
G_SPAWN_SEARCH_PATH,
NULL,
NULL,
&locate_pointer_pid,
&error);
locate_pointer_spawned = (error == NULL);
}
if (error) {
settings_mouse->set(KEY_MOUSE_LOCATE_POINTER,false);
g_error_free (error);
}
g_strfreev (args);
} else if (locate_pointer_spawned) {
kill (locate_pointer_pid, SIGHUP);
g_spawn_close_pid (locate_pointer_pid);
locate_pointer_spawned = FALSE;
}
}
void MouseManager::SetMouseWheelSpeed (int speed)
{
if(speed <= 0 )
return;
GPid pid;
QDir dir;
QString FilePath = dir.homePath() + "/.imwheelrc";
QFile file;
int delay = 240000 / speed;
QString date = QString("\".*\"\n"
"Control_L, Up, Control_L|Button4\n"
"Control_R, Up, Control_R|Button4\n"
"Control_L, Down, Control_L|Button5\n"
"Control_R, Down, Control_R|Button5\n"
"Shift_L, Up, Shift_L|Button4\n"
"Shift_R, Up, Shift_R|Button4\n"
"Shift_L, Down, Shift_L|Button5\n"
"Shift_R, Down, Shift_R|Button5\n"
"None, Up, Button4, %1, 0, %2\n"
"None, Down, Button5, %3, 0, %4\n")
.arg(speed).arg(delay).arg(speed).arg(delay);
file.setFileName(FilePath);
if(file.open(QIODevice::WriteOnly | QIODevice::Text)){
file.write(date.toLatin1().data());
}
GError *error = NULL;
char **args;
int argc;
if (imwheelSpawned){
QProcess::execute("killall imwheel");
imwheelSpawned = false;
}
QString str = "/usr/bin/imwheel";
if( g_shell_parse_argv (str.toLatin1().data(), &argc, &args, NULL)){
g_spawn_async (g_get_home_dir (),
args,
NULL,
G_SPAWN_SEARCH_PATH,
NULL,
NULL,
&pid,
&error);
imwheelSpawned = (error == NULL);
}
file.close();
g_strfreev (args);
}
void MouseManager::MouseCallback (QString keys)
{
if (keys.compare(QString::fromLocal8Bit(KEY_LEFT_HANDED))==0){
bool mouse_left_handed = settings_mouse->get(keys).toBool();
bool touchpad_left_handed = GetTouchpadHandedness (mouse_left_handed);
SetLeftHandedAll (mouse_left_handed, touchpad_left_handed);
} else if ((keys.compare(QString::fromLocal8Bit(KEY_MOTION_ACCELERATION))==0) ||
(keys.compare(QString::fromLocal8Bit(KEY_MOTION_THRESHOLD))==0) ||
(keys.compare(QString::fromLocal8Bit(KEY_MOUSE_ACCEL)) == 0)){
SetMotionAll ();
} else if (keys.compare(QString::fromLocal8Bit(KEY_MIDDLE_BUTTON_EMULATION))==0){
SetMiddleButtonAll (settings_mouse->get(keys).toBool());
} else if (keys.compare(QString::fromLocal8Bit(KEY_MOUSE_LOCATE_POINTER))==0){
SetLocatePointer (settings_mouse->get(keys).toBool());
} else if(keys.compare(QString::fromLocal8Bit(KEY_MOUSE_WHEEL_SPEED)) == 0 ) {
SetMouseWheelSpeed (settings_mouse->get(keys).toInt());
}
}
gboolean have_program_in_path (const char *name)
{
char *path;
bool result;
path = g_find_program_in_path (name);
result = (path != NULL);
g_free (path);
return result;
}
void MouseManager::SetDisableWTypingSynaptics (bool state)
{
if (state && touchpad_is_present ()) {
GError *error = NULL;
char **args;
int argc;
QString cmd = "syndaemon -i 0.5 -K -R";
if (syndaemon_spawned)
return;
if (!have_program_in_path ("syndaemon"))
return;
if (g_shell_parse_argv (cmd.toLatin1().data(), &argc, &args, NULL)) {
g_spawn_async (g_get_home_dir (),
args,
NULL,
G_SPAWN_SEARCH_PATH,
NULL,
NULL,
&syndaemon_pid,
&error);
syndaemon_spawned = (error == NULL);
}
if (error) {
settings_touchpad->set(KEY_TOUCHPAD_DISABLE_W_TYPING,false);
g_error_free (error);
}
g_strfreev (args);
} else if (syndaemon_spawned)
{
kill (syndaemon_pid, SIGHUP);
g_spawn_close_pid (syndaemon_pid);
syndaemon_spawned = FALSE;
}
}
void touchpad_set_bool (XDeviceInfo *device_info,
const char *property_name,
int property_index,
bool enabled)
{
XDevice *device;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
property_set_bool (device_info, device, property_name, property_index, enabled);
try {
XCloseDevice (QX11Info::display(), device);
} catch (int x) {
qWarning("%s:error",__func__);
}
}
void MouseManager::SetDisableWTypingLibinput (bool state)
{
XDeviceInfo *device_info = nullptr;
int n_devices;
int i;
/* This is only called once for synaptics but for libinput
* we need to loop through the list of devices
*/
device_info = XListInputDevices (QX11Info::display(), &n_devices);
if(!device_info){
qWarning("SetDisableWTypingLibinput: device_info is null");
return;
}
for (i = 0; i < n_devices; i++) {
touchpad_set_bool (&device_info[i], "libinput Disable While Typing Enabled", 0, state);
}
if (device_info != NULL)
XFreeDeviceList (device_info);
}
void MouseManager::SetDisableWTyping (bool state)
{
if (property_from_name ("Synaptics Off"))
SetDisableWTypingSynaptics (state);
if (property_from_name ("libinput Disable While Typing Enabled"))
SetDisableWTypingLibinput (state);
}
static void
set_tap_to_click_libinput (XDeviceInfo *device_info, bool state)
{
touchpad_set_bool (device_info, "libinput Tapping Enabled", 0, state);
}
static void
set_tap_to_click (XDeviceInfo *device_info, bool state, bool left_handed,
int one_finger_tap, int two_finger_tap, int three_finger_tap)
{
if (property_from_name ("Synaptics Tap Action"))
set_tap_to_click_synaptics (device_info, state, left_handed,
one_finger_tap, two_finger_tap, three_finger_tap);
if (property_from_name ("libinput Tapping Enabled"))
set_tap_to_click_libinput (device_info, state);
}
void MouseManager::SetTapToClickAll ()
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
bool state = settings_touchpad->get(KEY_TOUCHPAD_TAP_TO_CLICK).toBool();
bool left_handed = GetTouchpadHandedness (settings_mouse->get(KEY_LEFT_HANDED).toBool());
int one_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_ONE_FINGER_TAP).toInt();
int two_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_TWO_FINGER_TAP).toInt();
int three_finger_tap = settings_touchpad->get(KEY_TOUCHPAD_THREE_FINGER_TAP).toInt();
for (i = 0; i < numdevices; i++) {
set_tap_to_click (&devicelist[i], state, left_handed, one_finger_tap, two_finger_tap, three_finger_tap);
}
XFreeDeviceList (devicelist);
}
static void set_scrolling_synaptics (XDeviceInfo *device_info,
QGSettings *settings)
{
touchpad_set_bool (device_info, "Synaptics Edge Scrolling", 0, settings->get(KEY_VERT_EDGE_SCROLL).toBool());
touchpad_set_bool (device_info, "Synaptics Edge Scrolling", 1, settings->get(KEY_HORIZ_EDGE_SCROLL).toBool());
touchpad_set_bool (device_info, "Synaptics Two-Finger Scrolling", 0, settings->get(KEY_VERT_TWO_FINGER_SCROLL).toBool());
touchpad_set_bool (device_info, "Synaptics Two-Finger Scrolling", 1, settings->get(KEY_HORIZ_TWO_FINGER_SCROLL).toBool());
}
static void set_scrolling_libinput (XDeviceInfo *device_info,
QGSettings *settings)
{
XDevice *device;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char *data;
Atom prop, type;
bool want_edge, want_2fg;
bool want_horiz;
Display *display = QX11Info::display();
prop = property_from_name ("libinput Scroll Method Enabled");
if (!prop)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
want_2fg = settings->get(KEY_VERT_TWO_FINGER_SCROLL).toBool();
want_edge = settings->get(KEY_VERT_EDGE_SCROLL).toBool();
/* libinput only allows for one scroll method at a time.
* If both are set, pick 2fg scrolling.
*/
if (want_2fg)
want_edge = false;
qDebug ("setting scroll method on %s", device_info->name);
try {
rc = XGetDeviceProperty (display, device, prop, 0, 2,
False, XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 8 && nitems >= 3) {
data[0] = want_2fg;
data[1] = want_edge;
XChangeDeviceProperty (display, device,
prop, XA_INTEGER, 8, PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting scroll method on \"%s\"", device_info->name);
}
/* Horizontal scrolling is handled by xf86-input-libinput and
* there's only one bool. Pick the one matching the scroll method
* we picked above.
*/
if (want_2fg)
want_horiz = settings->get(KEY_HORIZ_TWO_FINGER_SCROLL).toBool();
else if (want_edge)
want_horiz = settings->get(KEY_HORIZ_EDGE_SCROLL).toBool();
else
return;
touchpad_set_bool (device_info, "libinput Horizontal Scroll Enabled", 0, want_horiz);
}
static void set_scrolling (XDeviceInfo *device_info,
QGSettings *settings)
{
if (property_from_name ("Synaptics Edge Scrolling"))
set_scrolling_synaptics (device_info, settings);
if (property_from_name ("libinput Scroll Method Enabled"))
set_scrolling_libinput (device_info, settings);
}
void SetScrollingAll (QGSettings *settings)
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
for (i = 0; i < numdevices; i++) {
set_scrolling (&devicelist[i], settings);
}
XFreeDeviceList (devicelist);
}
void set_natural_scroll_synaptics (XDeviceInfo *device_info,
bool natural_scroll)
{
XDevice *device;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char* data;
long *ptr;
Atom prop, type;
Display *display = QX11Info::display();
prop = property_from_name ("Synaptics Scrolling Distance");
if (!prop)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
qDebug ("Trying to set %s for \"%s\"",
natural_scroll ? "natural (reverse) scroll" : "normal scroll",
device_info->name);
try {
rc = XGetDeviceProperty (display , device, prop, 0, 2,
False, XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 32 && nitems >= 2) {
ptr = (glong *) data;
if (natural_scroll) {
ptr[0] = -abs(ptr[0]);
ptr[1] = -abs(ptr[1]);
} else {
ptr[0] = abs(ptr[0]);
ptr[1] = abs(ptr[1]);
}
XChangeDeviceProperty (display, device, prop,
XA_INTEGER, 32, PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting natural scroll on \"%s\"", device_info->name);
}
}
void set_natural_scroll_libinput (XDeviceInfo *device_info,
bool natural_scroll)
{
qDebug ("Trying to set %s for \"%s\"",
natural_scroll ? "natural (reverse) scroll" : "normal scroll",
device_info->name);
touchpad_set_bool (device_info, "libinput Natural Scrolling Enabled",
0, natural_scroll);
}
void set_natural_scroll (XDeviceInfo *device_info,
bool natural_scroll)
{
if (property_from_name ("Synaptics Scrolling Distance"))
set_natural_scroll_synaptics (device_info, natural_scroll);
if (property_from_name ("libinput Natural Scrolling Enabled"))
set_natural_scroll_libinput (device_info, natural_scroll);
}
void MouseManager::SetNaturalScrollAll ()
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
bool natural_scroll = settings_touchpad->get(KEY_TOUCHPAD_NATURAL_SCROLL).toBool();
for (i = 0; i < numdevices; i++) {
set_natural_scroll (&devicelist[i], natural_scroll);
}
XFreeDeviceList (devicelist);
}
void set_touchpad_enabled (XDeviceInfo *device_info,
bool state)
{
XDevice *device;
Atom prop_enabled;
unsigned char data = state;
Display *display = gdk_x11_get_default_xdisplay ();//QX11Info::display();//
prop_enabled = property_from_name ("Device Enabled");
if (!prop_enabled)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
try {
XChangeDeviceProperty (display, device,
prop_enabled, XA_INTEGER, 8,
PropModeReplace, &data, 1);
XCloseDevice (display, device);
gdk_display_flush (gdk_display_get_default());
} catch (int x) {
qWarning("Error %s device \"%s\"",
(state) ? "enabling" : "disabling",
device_info->name);
}
}
void SetTouchpadEnabledAll (bool state)
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices ( QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
for (i = 0; i < numdevices; i++) {
set_touchpad_enabled (&devicelist[i], state);
}
XFreeDeviceList (devicelist);
}
bool SetDisbleTouchpad(XDeviceInfo *device_info,
QGSettings *settings)
{
QString name;
bool state;
name = device_info->name;
bool Pmouse = name.contains("Mouse", Qt::CaseInsensitive);
bool Pusb = name.contains("USB", Qt::CaseInsensitive);
if(Pmouse && Pusb){
state = settings->get(KEY_TOUCHPAD_DISBLE_O_E_MOUSE).toBool();
if(state){
settings->set(KEY_TOUCHPAD_ENABLED, false);
return true;
}else {
settings->set(KEY_TOUCHPAD_ENABLED, true);
return true;
}
}
return false;
}
void SetPlugMouseDisbleTouchpad(QGSettings *settings)
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
for (i = 0; i < numdevices; i++) {
if(SetDisbleTouchpad (&devicelist[i], settings))
break;
}
XFreeDeviceList (devicelist);
}
void SetTouchpadDoubleClick(XDeviceInfo *device_info, bool state)
{
XDevice *device;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char* data;
Atom prop, type;
Display *display = gdk_x11_get_default_xdisplay ();//QX11Info::display();
prop = property_from_name ("Synaptics Gestures");
if (!prop)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
qDebug ("Trying to set for \"%s\"", device_info->name);
try {
rc = XGetDeviceProperty (display , device, prop, 0, 1,
False, XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 8 && nitems >= 1) {
if(state)
data[0]=1;
else
data[0]=0;
XChangeDeviceProperty (display, device, prop,
XA_INTEGER, 8, PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting natural scroll on \"%s\"", device_info->name);
}
}
void SetTouchpadDoubleClickAll(bool state)
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
for (i = 0; i < numdevices; i++) {
SetTouchpadDoubleClick (&devicelist[i], state);
}
XFreeDeviceList (devicelist);
}
//设置关闭右下角菜单
void MouseManager::SetBottomRightClickMenu(XDeviceInfo *device_info, bool state)
{
XDevice *device;
int format, rc;
unsigned long nitems, bytes_after;
unsigned char* data;
long *ptr;
Atom prop, type;
Display *display = gdk_x11_get_default_xdisplay ();//QX11Info::display();
prop = property_from_name ("Synaptics Soft Button Areas");
if (!prop)
return;
device = device_is_touchpad (device_info);
if (device == NULL) {
return;
}
qDebug ("Trying to set for \"%s\"", device_info->name);
try {
rc = XGetDeviceProperty (display , device, prop, 0, 8,
False, XA_INTEGER, &type, &format, &nitems,
&bytes_after, &data);
if (rc == Success && type == XA_INTEGER && format == 32 && nitems >= 3) {
ptr = (long *)data;
if(ptr[0] != 0){
mAreaLeft = ptr[0];
mAreaTop = ptr[2];
}
if (state) {
ptr[0] = mAreaLeft;
ptr[2] = mAreaTop;
} else {
ptr[0] = 0;
ptr[2] = 0;
}
XChangeDeviceProperty (display, device, prop,
XA_INTEGER, 32, PropModeReplace, data, nitems);
}
if (rc == Success)
XFree (data);
XCloseDevice (display, device);
} catch (int x) {
qWarning("Error in setting natural scroll on \"%s\"", device_info->name);
}
}
void MouseManager::SetBottomRightConrnerClickMenu(bool state)
{
int numdevices, i;
XDeviceInfo *devicelist = XListInputDevices (QX11Info::display(), &numdevices);
if (devicelist == NULL)
return;
for (i = 0; i < numdevices; i++) {
SetBottomRightClickMenu(&devicelist[i], state);
}
XFreeDeviceList (devicelist);
}
void MouseManager::TouchpadCallback (QString keys)
{
if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_DISABLE_W_TYPING))==0) {
SetDisableWTyping (settings_touchpad->get(keys).toBool()); //设置打字时禁用触摸板
} else if (keys.compare(QString::fromLocal8Bit(KEY_LEFT_HANDED))== 0) {
bool mouse_left_handed = settings_mouse->get(keys).toBool();
bool touchpad_left_handed = GetTouchpadHandedness (mouse_left_handed);
SetLeftHandedAll (mouse_left_handed, touchpad_left_handed); //设置左右手
} else if ((keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_TAP_TO_CLICK)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_ONE_FINGER_TAP)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_TWO_FINGER_TAP)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_THREE_FINGER_TAP))== 0)) {
SetTapToClickAll (); //设置多指手势
} else if ((keys.compare(QString::fromLocal8Bit(KEY_VERT_EDGE_SCROLL)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_HORIZ_EDGE_SCROLL)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_VERT_TWO_FINGER_SCROLL)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_HORIZ_TWO_FINGER_SCROLL)) == 0)) {
SetScrollingAll (settings_touchpad); //设置滚动
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_NATURAL_SCROLL)) == 0) {
SetNaturalScrollAll (); //设置上移下滚或上移上滚
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_ENABLED)) == 0) {
SetTouchpadEnabledAll (settings_touchpad->get(keys).toBool());//设置触摸板开关
} else if ((keys.compare(QString::fromLocal8Bit(KEY_MOTION_ACCELERATION)) == 0)
|| (keys.compare(QString::fromLocal8Bit(KEY_MOTION_THRESHOLD)) == 0)) {
SetMotionAll (); //设置鼠标速度
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_DISBLE_O_E_MOUSE)) == 0) {
SetPlugMouseDisbleTouchpad(settings_touchpad); //设置插入鼠标时禁用触摸板
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_DOUBLE_CLICK_DRAG)) == 0){
SetTouchpadDoubleClickAll(settings_touchpad->get(KEY_TOUCHPAD_DOUBLE_CLICK_DRAG).toBool());//设置轻点击两次拖动打开关闭
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_BOTTOM_R_C_CLICK_M)) == 0){
SetBottomRightConrnerClickMenu(settings_touchpad->get(KEY_TOUCHPAD_BOTTOM_R_C_CLICK_M).toBool());//打开关闭右下角点击弹出菜单
} else if (keys.compare(QString::fromLocal8Bit(KEY_TOUCHPAD_MOUSE_SENSITVITY)) == 0){
}
}
void MouseManager::SetMouseSettings ()
{
bool mouse_left_handed = settings_mouse->get(KEY_LEFT_HANDED).toBool();
bool touchpad_left_handed = GetTouchpadHandedness (mouse_left_handed);
SetLeftHandedAll (mouse_left_handed, touchpad_left_handed);
SetMotionAll ();
SetMiddleButtonAll (settings_mouse->get(KEY_MIDDLE_BUTTON_EMULATION).toBool());
SetDisableWTyping (settings_touchpad->get(KEY_TOUCHPAD_DISABLE_W_TYPING).toBool());
SetTapToClickAll ();
SetScrollingAll (settings_touchpad);
SetNaturalScrollAll ();
SetTouchpadEnabledAll (settings_touchpad->get(KEY_TOUCHPAD_ENABLED).toBool());
SetPlugMouseDisbleTouchpad(settings_touchpad);
SetTouchpadDoubleClickAll(settings_touchpad->get(KEY_TOUCHPAD_DOUBLE_CLICK_DRAG).toBool());
SetBottomRightConrnerClickMenu(settings_touchpad->get(KEY_TOUCHPAD_BOTTOM_R_C_CLICK_M).toBool());
}
GdkFilterReturn devicepresence_filter (GdkXEvent *xevent,
GdkEvent *event,
gpointer data)
{
XEvent *xev = (XEvent *) xevent;
XEventClass class_presence;
int xi_presence;
MouseManager * manager = (MouseManager *) data;
DevicePresence (gdk_x11_get_default_xdisplay (), xi_presence, class_presence);
if (xev->type == xi_presence)
{
XDevicePresenceNotifyEvent *dpn = (XDevicePresenceNotifyEvent *) xev;
if (dpn->devchange == DeviceEnabled)
manager->SetMouseSettings ();
}
return GDK_FILTER_CONTINUE;
}
void MouseManager::SetDevicepresenceHandler ()
{
Display *display;
XEventClass class_presence;
int xi_presence;
display = QX11Info::display();
gdk_x11_display_error_trap_push (gdk_display_get_default());
DevicePresence (display, xi_presence, class_presence);
XSelectExtensionEvent (display,
RootWindow (display, DefaultScreen (display)),
&class_presence, 1);
gdk_display_flush (gdk_display_get_default());
if (!gdk_x11_display_error_trap_pop (gdk_display_get_default()))
gdk_window_add_filter (NULL, devicepresence_filter,this);
}
void MouseManager::MouseManagerIdleCb()
{
time->stop();
QObject::connect(settings_mouse,SIGNAL(changed(QString)),
this,SLOT(MouseCallback(QString)));
QObject::connect(settings_touchpad,SIGNAL(changed(QString)),
this,SLOT(TouchpadCallback(QString)));
syndaemon_spawned = FALSE;
SetDevicepresenceHandler ();
SetMouseSettings ();
SetLocatePointer (settings_mouse->get(KEY_MOUSE_LOCATE_POINTER).toBool());
}