refact: embed crate mouce, uinput (#8836)
Signed-off-by: fufesou <linlong1266@gmail.com>
This commit is contained in:
parent
541d9c6b86
commit
c0e9445602
9
Cargo.lock
generated
9
Cargo.lock
generated
@ -3777,14 +3777,6 @@ dependencies = [
|
|||||||
"windows-sys 0.48.0",
|
"windows-sys 0.48.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "mouce"
|
|
||||||
version = "0.2.1"
|
|
||||||
source = "git+https://github.com/rustdesk-org/mouce.git#177625a395cd8fa73964714d0039535cb9b47893"
|
|
||||||
dependencies = [
|
|
||||||
"glob",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "muda"
|
name = "muda"
|
||||||
version = "0.13.5"
|
version = "0.13.5"
|
||||||
@ -5514,7 +5506,6 @@ dependencies = [
|
|||||||
"libpulse-simple-binding",
|
"libpulse-simple-binding",
|
||||||
"mac_address",
|
"mac_address",
|
||||||
"magnum-opus",
|
"magnum-opus",
|
||||||
"mouce",
|
|
||||||
"num_cpus",
|
"num_cpus",
|
||||||
"objc",
|
"objc",
|
||||||
"objc_id",
|
"objc_id",
|
||||||
|
@ -152,7 +152,6 @@ psimple = { package = "libpulse-simple-binding", version = "2.27" }
|
|||||||
pulse = { package = "libpulse-binding", version = "2.27" }
|
pulse = { package = "libpulse-binding", version = "2.27" }
|
||||||
rust-pulsectl = { git = "https://github.com/open-trade/pulsectl" }
|
rust-pulsectl = { git = "https://github.com/open-trade/pulsectl" }
|
||||||
async-process = "1.7"
|
async-process = "1.7"
|
||||||
mouce = { git="https://github.com/rustdesk-org/mouce.git" }
|
|
||||||
evdev = { git="https://github.com/rustdesk-org/evdev" }
|
evdev = { git="https://github.com/rustdesk-org/evdev" }
|
||||||
dbus = "0.9"
|
dbus = "0.9"
|
||||||
dbus-crossroads = "0.5"
|
dbus-crossroads = "0.5"
|
||||||
|
@ -4,7 +4,11 @@ use evdev::{
|
|||||||
uinput::{VirtualDevice, VirtualDeviceBuilder},
|
uinput::{VirtualDevice, VirtualDeviceBuilder},
|
||||||
AttributeSet, EventType, InputEvent,
|
AttributeSet, EventType, InputEvent,
|
||||||
};
|
};
|
||||||
use hbb_common::{allow_err, bail, log, tokio::{self, runtime::Runtime}, ResultType};
|
use hbb_common::{
|
||||||
|
allow_err, bail, log,
|
||||||
|
tokio::{self, runtime::Runtime},
|
||||||
|
ResultType,
|
||||||
|
};
|
||||||
|
|
||||||
static IPC_CONN_TIMEOUT: u64 = 1000;
|
static IPC_CONN_TIMEOUT: u64 = 1000;
|
||||||
static IPC_REQUEST_TIMEOUT: u64 = 1000;
|
static IPC_REQUEST_TIMEOUT: u64 = 1000;
|
||||||
@ -34,7 +38,10 @@ pub mod client {
|
|||||||
fn send_get_key_state(&mut self, data: Data) -> ResultType<bool> {
|
fn send_get_key_state(&mut self, data: Data) -> ResultType<bool> {
|
||||||
self.rt.block_on(self.conn.send(&data))?;
|
self.rt.block_on(self.conn.send(&data))?;
|
||||||
|
|
||||||
match self.rt.block_on(self.conn.next_timeout(IPC_REQUEST_TIMEOUT)) {
|
match self
|
||||||
|
.rt
|
||||||
|
.block_on(self.conn.next_timeout(IPC_REQUEST_TIMEOUT))
|
||||||
|
{
|
||||||
Ok(Some(Data::KeyboardResponse(ipc::DataKeyboardResponse::GetKeyState(state)))) => {
|
Ok(Some(Data::KeyboardResponse(ipc::DataKeyboardResponse::GetKeyState(state)))) => {
|
||||||
Ok(state)
|
Ok(state)
|
||||||
}
|
}
|
||||||
@ -171,7 +178,6 @@ pub mod client {
|
|||||||
pub mod service {
|
pub mod service {
|
||||||
use super::*;
|
use super::*;
|
||||||
use hbb_common::lazy_static;
|
use hbb_common::lazy_static;
|
||||||
use mouce::MouseActions;
|
|
||||||
use std::{collections::HashMap, sync::Mutex};
|
use std::{collections::HashMap, sync::Mutex};
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
@ -389,7 +395,7 @@ pub mod service {
|
|||||||
} else {
|
} else {
|
||||||
match key {
|
match key {
|
||||||
enigo::Key::Layout(c) => {
|
enigo::Key::Layout(c) => {
|
||||||
if let Some((k,is_shift)) = KEY_MAP_LAYOUT.get(&c) {
|
if let Some((k, is_shift)) = KEY_MAP_LAYOUT.get(&c) {
|
||||||
log::trace!("mapkey {:?}, get {:?}", &key, k);
|
log::trace!("mapkey {:?}, get {:?}", &key, k);
|
||||||
return Ok((k.clone(), is_shift.clone()));
|
return Ok((k.clone(), is_shift.clone()));
|
||||||
}
|
}
|
||||||
@ -431,7 +437,8 @@ pub mod service {
|
|||||||
DataKeyboard::KeyDown(key) => {
|
DataKeyboard::KeyDown(key) => {
|
||||||
if let Ok((k, is_shift)) = map_key(key) {
|
if let Ok((k, is_shift)) = map_key(key) {
|
||||||
if is_shift {
|
if is_shift {
|
||||||
let down_event = InputEvent::new(EventType::KEY, evdev::Key::KEY_LEFTSHIFT.code(), 1);
|
let down_event =
|
||||||
|
InputEvent::new(EventType::KEY, evdev::Key::KEY_LEFTSHIFT.code(), 1);
|
||||||
allow_err!(keyboard.emit(&[down_event]));
|
allow_err!(keyboard.emit(&[down_event]));
|
||||||
}
|
}
|
||||||
let down_event = InputEvent::new(EventType::KEY, k.code(), 1);
|
let down_event = InputEvent::new(EventType::KEY, k.code(), 1);
|
||||||
@ -504,7 +511,7 @@ pub mod service {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_mouse(mouse: &mut mouce::nix::UInputMouseManager, data: &DataMouse) {
|
fn handle_mouse(mouse: &mut mouce::UInputMouseManager, data: &DataMouse) {
|
||||||
log::trace!("handle_mouse {:?}", &data);
|
log::trace!("handle_mouse {:?}", &data);
|
||||||
match data {
|
match data {
|
||||||
DataMouse::MoveTo(x, y) => {
|
DataMouse::MoveTo(x, y) => {
|
||||||
@ -515,9 +522,9 @@ pub mod service {
|
|||||||
}
|
}
|
||||||
DataMouse::Down(button) => {
|
DataMouse::Down(button) => {
|
||||||
let btn = match button {
|
let btn = match button {
|
||||||
enigo::MouseButton::Left => mouce::common::MouseButton::Left,
|
enigo::MouseButton::Left => mouce::MouseButton::Left,
|
||||||
enigo::MouseButton::Middle => mouce::common::MouseButton::Middle,
|
enigo::MouseButton::Middle => mouce::MouseButton::Middle,
|
||||||
enigo::MouseButton::Right => mouce::common::MouseButton::Right,
|
enigo::MouseButton::Right => mouce::MouseButton::Right,
|
||||||
_ => {
|
_ => {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -526,9 +533,9 @@ pub mod service {
|
|||||||
}
|
}
|
||||||
DataMouse::Up(button) => {
|
DataMouse::Up(button) => {
|
||||||
let btn = match button {
|
let btn = match button {
|
||||||
enigo::MouseButton::Left => mouce::common::MouseButton::Left,
|
enigo::MouseButton::Left => mouce::MouseButton::Left,
|
||||||
enigo::MouseButton::Middle => mouce::common::MouseButton::Middle,
|
enigo::MouseButton::Middle => mouce::MouseButton::Middle,
|
||||||
enigo::MouseButton::Right => mouce::common::MouseButton::Right,
|
enigo::MouseButton::Right => mouce::MouseButton::Right,
|
||||||
_ => {
|
_ => {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -537,9 +544,9 @@ pub mod service {
|
|||||||
}
|
}
|
||||||
DataMouse::Click(button) => {
|
DataMouse::Click(button) => {
|
||||||
let btn = match button {
|
let btn = match button {
|
||||||
enigo::MouseButton::Left => mouce::common::MouseButton::Left,
|
enigo::MouseButton::Left => mouce::MouseButton::Left,
|
||||||
enigo::MouseButton::Middle => mouce::common::MouseButton::Middle,
|
enigo::MouseButton::Middle => mouce::MouseButton::Middle,
|
||||||
enigo::MouseButton::Right => mouce::common::MouseButton::Right,
|
enigo::MouseButton::Right => mouce::MouseButton::Right,
|
||||||
_ => {
|
_ => {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -553,9 +560,9 @@ pub mod service {
|
|||||||
let mut length = *length;
|
let mut length = *length;
|
||||||
|
|
||||||
let scroll = if length < 0 {
|
let scroll = if length < 0 {
|
||||||
mouce::common::ScrollDirection::Up
|
mouce::ScrollDirection::Up
|
||||||
} else {
|
} else {
|
||||||
mouce::common::ScrollDirection::Down
|
mouce::ScrollDirection::Down
|
||||||
};
|
};
|
||||||
|
|
||||||
if length < 0 {
|
if length < 0 {
|
||||||
@ -621,7 +628,7 @@ pub mod service {
|
|||||||
rng_y.0,
|
rng_y.0,
|
||||||
rng_y.1
|
rng_y.1
|
||||||
);
|
);
|
||||||
let mut mouse = match mouce::Mouse::new_uinput(rng_x, rng_y) {
|
let mut mouse = match mouce::UInputMouseManager::new(rng_x, rng_y) {
|
||||||
Ok(mouse) => mouse,
|
Ok(mouse) => mouse,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Failed to create mouse, {}", e);
|
log::error!("Failed to create mouse, {}", e);
|
||||||
@ -650,7 +657,7 @@ pub mod service {
|
|||||||
rng_y.0,
|
rng_y.0,
|
||||||
rng_y.1
|
rng_y.1
|
||||||
);
|
);
|
||||||
mouse = match mouce::Mouse::new_uinput(rng_x, rng_y) {
|
mouse = match mouce::UInputMouseManager::new(rng_x, rng_y) {
|
||||||
Ok(mouse) => mouse,
|
Ok(mouse) => mouse,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
log::error!("Failed to create mouse, {}", e);
|
log::error!("Failed to create mouse, {}", e);
|
||||||
@ -761,3 +768,346 @@ pub mod service {
|
|||||||
log::info!("stop uinput control service");
|
log::info!("stop uinput control service");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://github.com/emrebicer/mouce
|
||||||
|
mod mouce {
|
||||||
|
use std::{
|
||||||
|
fs::File,
|
||||||
|
io::{Error, ErrorKind, Result},
|
||||||
|
mem::size_of,
|
||||||
|
os::{
|
||||||
|
raw::{c_char, c_int, c_long, c_uint, c_ulong, c_ushort},
|
||||||
|
unix::{fs::OpenOptionsExt, io::AsRawFd},
|
||||||
|
},
|
||||||
|
thread,
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const O_NONBLOCK: c_int = 2048;
|
||||||
|
|
||||||
|
/// ioctl and uinput definitions
|
||||||
|
const UI_ABS_SETUP: c_ulong = 1075598596;
|
||||||
|
const UI_SET_EVBIT: c_ulong = 1074025828;
|
||||||
|
const UI_SET_KEYBIT: c_ulong = 1074025829;
|
||||||
|
const UI_SET_RELBIT: c_ulong = 1074025830;
|
||||||
|
const UI_SET_ABSBIT: c_ulong = 1074025831;
|
||||||
|
const UI_DEV_SETUP: c_ulong = 1079792899;
|
||||||
|
const UI_DEV_CREATE: c_ulong = 21761;
|
||||||
|
const UI_DEV_DESTROY: c_uint = 21762;
|
||||||
|
|
||||||
|
pub const EV_KEY: c_int = 0x01;
|
||||||
|
pub const EV_REL: c_int = 0x02;
|
||||||
|
pub const EV_ABS: c_int = 0x03;
|
||||||
|
pub const REL_X: c_uint = 0x00;
|
||||||
|
pub const REL_Y: c_uint = 0x01;
|
||||||
|
pub const ABS_X: c_uint = 0x00;
|
||||||
|
pub const ABS_Y: c_uint = 0x01;
|
||||||
|
pub const REL_WHEEL: c_uint = 0x08;
|
||||||
|
pub const REL_HWHEEL: c_uint = 0x06;
|
||||||
|
pub const BTN_LEFT: c_int = 0x110;
|
||||||
|
pub const BTN_RIGHT: c_int = 0x111;
|
||||||
|
pub const BTN_MIDDLE: c_int = 0x112;
|
||||||
|
pub const BTN_SIDE: c_int = 0x113;
|
||||||
|
pub const BTN_EXTRA: c_int = 0x114;
|
||||||
|
pub const BTN_FORWARD: c_int = 0x115;
|
||||||
|
pub const BTN_BACK: c_int = 0x116;
|
||||||
|
pub const BTN_TASK: c_int = 0x117;
|
||||||
|
const SYN_REPORT: c_int = 0x00;
|
||||||
|
const EV_SYN: c_int = 0x00;
|
||||||
|
const BUS_USB: c_ushort = 0x03;
|
||||||
|
|
||||||
|
/// uinput types
|
||||||
|
#[repr(C)]
|
||||||
|
struct UInputSetup {
|
||||||
|
id: InputId,
|
||||||
|
name: [c_char; UINPUT_MAX_NAME_SIZE],
|
||||||
|
ff_effects_max: c_ulong,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
struct InputId {
|
||||||
|
bustype: c_ushort,
|
||||||
|
vendor: c_ushort,
|
||||||
|
product: c_ushort,
|
||||||
|
version: c_ushort,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct InputEvent {
|
||||||
|
pub time: TimeVal,
|
||||||
|
pub r#type: c_ushort,
|
||||||
|
pub code: c_ushort,
|
||||||
|
pub value: c_int,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct TimeVal {
|
||||||
|
pub tv_sec: c_ulong,
|
||||||
|
pub tv_usec: c_ulong,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct UinputAbsSetup {
|
||||||
|
pub code: c_ushort,
|
||||||
|
pub absinfo: InputAbsinfo,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct InputAbsinfo {
|
||||||
|
pub value: c_int,
|
||||||
|
pub minimum: c_int,
|
||||||
|
pub maximum: c_int,
|
||||||
|
pub fuzz: c_int,
|
||||||
|
pub flat: c_int,
|
||||||
|
pub resolution: c_int,
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
fn ioctl(fd: c_int, request: c_ulong, ...) -> c_int;
|
||||||
|
fn write(fd: c_int, buf: *mut InputEvent, count: usize) -> c_long;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone)]
|
||||||
|
pub enum MouseButton {
|
||||||
|
Left,
|
||||||
|
Middle,
|
||||||
|
Side,
|
||||||
|
Extra,
|
||||||
|
Right,
|
||||||
|
Back,
|
||||||
|
Forward,
|
||||||
|
Task,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone)]
|
||||||
|
pub enum ScrollDirection {
|
||||||
|
Up,
|
||||||
|
Down,
|
||||||
|
Right,
|
||||||
|
Left,
|
||||||
|
}
|
||||||
|
|
||||||
|
const UINPUT_MAX_NAME_SIZE: usize = 80;
|
||||||
|
|
||||||
|
pub struct UInputMouseManager {
|
||||||
|
uinput_file: File,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UInputMouseManager {
|
||||||
|
pub fn new(rng_x: (i32, i32), rng_y: (i32, i32)) -> Result<Self> {
|
||||||
|
let manager = UInputMouseManager {
|
||||||
|
uinput_file: File::options()
|
||||||
|
.write(true)
|
||||||
|
.custom_flags(O_NONBLOCK)
|
||||||
|
.open("/dev/uinput")?,
|
||||||
|
};
|
||||||
|
let fd = manager.uinput_file.as_raw_fd();
|
||||||
|
unsafe {
|
||||||
|
// For press events (also needed for mouse movement)
|
||||||
|
ioctl(fd, UI_SET_EVBIT, EV_KEY);
|
||||||
|
ioctl(fd, UI_SET_KEYBIT, BTN_LEFT);
|
||||||
|
ioctl(fd, UI_SET_KEYBIT, BTN_RIGHT);
|
||||||
|
ioctl(fd, UI_SET_KEYBIT, BTN_MIDDLE);
|
||||||
|
|
||||||
|
// For mouse movement
|
||||||
|
ioctl(fd, UI_SET_EVBIT, EV_ABS);
|
||||||
|
ioctl(fd, UI_SET_ABSBIT, ABS_X);
|
||||||
|
ioctl(
|
||||||
|
fd,
|
||||||
|
UI_ABS_SETUP,
|
||||||
|
&UinputAbsSetup {
|
||||||
|
code: ABS_X as _,
|
||||||
|
absinfo: InputAbsinfo {
|
||||||
|
value: 0,
|
||||||
|
minimum: rng_x.0,
|
||||||
|
maximum: rng_x.1,
|
||||||
|
fuzz: 0,
|
||||||
|
flat: 0,
|
||||||
|
resolution: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
ioctl(fd, UI_SET_ABSBIT, ABS_Y);
|
||||||
|
ioctl(
|
||||||
|
fd,
|
||||||
|
UI_ABS_SETUP,
|
||||||
|
&UinputAbsSetup {
|
||||||
|
code: ABS_Y as _,
|
||||||
|
absinfo: InputAbsinfo {
|
||||||
|
value: 0,
|
||||||
|
minimum: rng_y.0,
|
||||||
|
maximum: rng_y.1,
|
||||||
|
fuzz: 0,
|
||||||
|
flat: 0,
|
||||||
|
resolution: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
ioctl(fd, UI_SET_EVBIT, EV_REL);
|
||||||
|
ioctl(fd, UI_SET_RELBIT, REL_X);
|
||||||
|
ioctl(fd, UI_SET_RELBIT, REL_Y);
|
||||||
|
ioctl(fd, UI_SET_RELBIT, REL_WHEEL);
|
||||||
|
ioctl(fd, UI_SET_RELBIT, REL_HWHEEL);
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut usetup = UInputSetup {
|
||||||
|
id: InputId {
|
||||||
|
bustype: BUS_USB,
|
||||||
|
// Random vendor and product
|
||||||
|
vendor: 0x2222,
|
||||||
|
product: 0x3333,
|
||||||
|
version: 0,
|
||||||
|
},
|
||||||
|
name: [0; UINPUT_MAX_NAME_SIZE],
|
||||||
|
ff_effects_max: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut device_bytes: Vec<c_char> = "mouce-library-fake-mouse"
|
||||||
|
.chars()
|
||||||
|
.map(|ch| ch as c_char)
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
// Fill the rest of the name buffer with empty chars
|
||||||
|
for _ in 0..UINPUT_MAX_NAME_SIZE - device_bytes.len() {
|
||||||
|
device_bytes.push('\0' as c_char);
|
||||||
|
}
|
||||||
|
|
||||||
|
usetup.name.copy_from_slice(&device_bytes);
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
ioctl(fd, UI_DEV_SETUP, &usetup);
|
||||||
|
ioctl(fd, UI_DEV_CREATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
// On UI_DEV_CREATE the kernel will create the device node for this
|
||||||
|
// device. We are inserting a pause here so that userspace has time
|
||||||
|
// to detect, initialize the new device, and can start listening to
|
||||||
|
// the event, otherwise it will not notice the event we are about to send.
|
||||||
|
thread::sleep(Duration::from_millis(300));
|
||||||
|
|
||||||
|
Ok(manager)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Write the given event to the uinput file
|
||||||
|
fn emit(&self, r#type: c_int, code: c_int, value: c_int) -> Result<()> {
|
||||||
|
let mut event = InputEvent {
|
||||||
|
time: TimeVal {
|
||||||
|
tv_sec: 0,
|
||||||
|
tv_usec: 0,
|
||||||
|
},
|
||||||
|
r#type: r#type as c_ushort,
|
||||||
|
code: code as c_ushort,
|
||||||
|
value,
|
||||||
|
};
|
||||||
|
let fd = self.uinput_file.as_raw_fd();
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
let count = size_of::<InputEvent>();
|
||||||
|
let written_bytes = write(fd, &mut event, count);
|
||||||
|
if written_bytes == -1 || written_bytes != count as c_long {
|
||||||
|
return Err(Error::new(
|
||||||
|
ErrorKind::Other,
|
||||||
|
format!("failed while trying to write to a file"),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Syncronize the device
|
||||||
|
fn syncronize(&self) -> Result<()> {
|
||||||
|
self.emit(EV_SYN, SYN_REPORT, 0)?;
|
||||||
|
// Give uinput some time to update the mouse location,
|
||||||
|
// otherwise it fails to move the mouse on release mode
|
||||||
|
// A delay of 1 milliseconds seems to be enough for it
|
||||||
|
thread::sleep(Duration::from_millis(1));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Move the mouse relative to the current position
|
||||||
|
fn move_relative_(&self, x: i32, y: i32) -> Result<()> {
|
||||||
|
// uinput does not move the mouse in pixels but uses `units`. I couldn't
|
||||||
|
// find information regarding to this uinput `unit`, but according to
|
||||||
|
// my findings 1 unit corresponds to exactly 2 pixels.
|
||||||
|
//
|
||||||
|
// To achieve the expected behavior; divide the parameters by 2
|
||||||
|
//
|
||||||
|
// This seems like there is a bug in this crate, but the
|
||||||
|
// behavior is the same on other projects that make use of
|
||||||
|
// uinput. e.g. `ydotool`. When you try to move your mouse,
|
||||||
|
// it will move 2x further pixels
|
||||||
|
self.emit(EV_REL, REL_X as c_int, (x as f32 / 2.).ceil() as c_int)?;
|
||||||
|
self.emit(EV_REL, REL_Y as c_int, (y as f32 / 2.).ceil() as c_int)?;
|
||||||
|
self.syncronize()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn map_btn(button: &MouseButton) -> c_int {
|
||||||
|
match button {
|
||||||
|
MouseButton::Left => BTN_LEFT,
|
||||||
|
MouseButton::Right => BTN_RIGHT,
|
||||||
|
MouseButton::Middle => BTN_MIDDLE,
|
||||||
|
MouseButton::Side => BTN_SIDE,
|
||||||
|
MouseButton::Extra => BTN_EXTRA,
|
||||||
|
MouseButton::Forward => BTN_FORWARD,
|
||||||
|
MouseButton::Back => BTN_BACK,
|
||||||
|
MouseButton::Task => BTN_TASK,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn move_to(&self, x: usize, y: usize) -> Result<()> {
|
||||||
|
// // For some reason, absolute mouse move events are not working on uinput
|
||||||
|
// // (as I understand those events are intended for touch events)
|
||||||
|
// //
|
||||||
|
// // As a work around solution; first set the mouse to top left, then
|
||||||
|
// // call relative move function to simulate an absolute move event
|
||||||
|
//self.move_relative(i32::MIN, i32::MIN)?;
|
||||||
|
//self.move_relative(x as i32, y as i32)
|
||||||
|
|
||||||
|
self.emit(EV_ABS, ABS_X as c_int, x as c_int)?;
|
||||||
|
self.emit(EV_ABS, ABS_Y as c_int, y as c_int)?;
|
||||||
|
self.syncronize()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn move_relative(&self, x_offset: i32, y_offset: i32) -> Result<()> {
|
||||||
|
self.move_relative_(x_offset, y_offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn press_button(&self, button: &MouseButton) -> Result<()> {
|
||||||
|
self.emit(EV_KEY, Self::map_btn(button), 1)?;
|
||||||
|
self.syncronize()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn release_button(&self, button: &MouseButton) -> Result<()> {
|
||||||
|
self.emit(EV_KEY, Self::map_btn(button), 0)?;
|
||||||
|
self.syncronize()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn click_button(&self, button: &MouseButton) -> Result<()> {
|
||||||
|
self.press_button(button)?;
|
||||||
|
self.release_button(button)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn scroll_wheel(&self, direction: &ScrollDirection) -> Result<()> {
|
||||||
|
let (code, scroll_value) = match direction {
|
||||||
|
ScrollDirection::Up => (REL_WHEEL, 1),
|
||||||
|
ScrollDirection::Down => (REL_WHEEL, -1),
|
||||||
|
ScrollDirection::Left => (REL_HWHEEL, -1),
|
||||||
|
ScrollDirection::Right => (REL_HWHEEL, 1),
|
||||||
|
};
|
||||||
|
self.emit(EV_REL, code as c_int, scroll_value)?;
|
||||||
|
self.syncronize()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for UInputMouseManager {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
let fd = self.uinput_file.as_raw_fd();
|
||||||
|
unsafe {
|
||||||
|
// Destroy the device, the file is closed automatically by the File module
|
||||||
|
ioctl(fd, UI_DEV_DESTROY as c_ulong);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user