rustdesk/src/server/input_service.rs

1011 lines
31 KiB
Rust
Raw Normal View History

2021-03-29 15:59:14 +08:00
use super::*;
#[cfg(target_os = "linux")]
2022-09-05 23:50:42 +08:00
use crate::common::IS_X11;
2021-03-29 15:59:14 +08:00
#[cfg(target_os = "macos")]
use dispatch::Queue;
use enigo::{Enigo, Key, KeyboardControllable, MouseButton, MouseControllable};
use hbb_common::{config::COMPRESS_LEVEL, get_time, protobuf::EnumOrUnknown};
2022-07-18 17:42:02 +08:00
use rdev::{simulate, EventType, Key as RdevKey};
2021-08-12 01:25:32 +08:00
use std::{
convert::TryFrom,
ops::Sub,
2021-08-12 01:25:32 +08:00
sync::atomic::{AtomicBool, Ordering},
time::Instant,
2021-08-12 01:25:32 +08:00
};
2021-03-29 15:59:14 +08:00
const INVALID_CURSOR_POS: i32 = i32::MIN;
2021-03-29 15:59:14 +08:00
#[derive(Default)]
struct StateCursor {
hcursor: u64,
cursor_data: Arc<Message>,
cached_cursor_data: HashMap<u64, Arc<Message>>,
}
impl super::service::Reset for StateCursor {
fn reset(&mut self) {
*self = Default::default();
crate::platform::reset_input_cache();
2022-03-07 16:19:10 +08:00
fix_key_down_timeout(true);
2021-03-29 15:59:14 +08:00
}
}
struct StatePos {
cursor_pos: (i32, i32),
}
impl Default for StatePos {
fn default() -> Self {
Self {
cursor_pos: (INVALID_CURSOR_POS, INVALID_CURSOR_POS),
}
}
}
2021-03-29 15:59:14 +08:00
impl super::service::Reset for StatePos {
fn reset(&mut self) {
self.cursor_pos = (INVALID_CURSOR_POS, INVALID_CURSOR_POS);
}
}
impl StatePos {
fn is_valid(&self) -> bool {
self.cursor_pos.0 != INVALID_CURSOR_POS && self.cursor_pos.1 != INVALID_CURSOR_POS
2021-03-29 15:59:14 +08:00
}
}
#[derive(Default, Clone, Copy)]
2021-03-29 15:59:14 +08:00
struct Input {
conn: i32,
time: i64,
x: i32,
y: i32,
2021-03-29 15:59:14 +08:00
}
2022-01-14 01:15:03 +08:00
const KEY_CHAR_START: u64 = 9999;
2021-03-29 15:59:14 +08:00
#[derive(Clone, Default)]
pub struct MouseCursorSub {
inner: ConnInner,
cached: HashMap<u64, Arc<Message>>,
}
impl From<ConnInner> for MouseCursorSub {
fn from(inner: ConnInner) -> Self {
Self {
inner,
cached: HashMap::new(),
}
}
}
impl Subscriber for MouseCursorSub {
#[inline]
fn id(&self) -> i32 {
self.inner.id()
}
#[inline]
fn send(&mut self, msg: Arc<Message>) {
if let Some(message::Union::CursorData(cd)) = &msg.union {
2021-03-29 15:59:14 +08:00
if let Some(msg) = self.cached.get(&cd.id) {
self.inner.send(msg.clone());
} else {
self.inner.send(msg.clone());
let mut tmp = Message::new();
// only send id out, require client side cache also
tmp.set_cursor_id(cd.id);
self.cached.insert(cd.id, Arc::new(tmp));
}
} else {
self.inner.send(msg);
}
}
}
pub const NAME_CURSOR: &'static str = "mouse_cursor";
pub const NAME_POS: &'static str = "mouse_pos";
pub type MouseCursorService = ServiceTmpl<MouseCursorSub>;
pub fn new_cursor() -> MouseCursorService {
let sp = MouseCursorService::new(NAME_CURSOR, true);
sp.repeat::<StateCursor, _>(33, run_cursor);
sp
}
pub fn new_pos() -> GenericService {
let sp = GenericService::new(NAME_POS, false);
sp.repeat::<StatePos, _>(33, run_pos);
sp
}
fn update_last_cursor_pos(x: i32, y: i32) {
let mut lock = LATEST_CURSOR_POS.lock().unwrap();
if lock.1 .0 != x || lock.1 .1 != y {
(lock.0, lock.1) = (Instant::now(), (x, y))
}
}
2021-03-29 15:59:14 +08:00
fn run_pos(sp: GenericService, state: &mut StatePos) -> ResultType<()> {
if let Some((x, y)) = crate::get_cursor_pos() {
update_last_cursor_pos(x, y);
if state.is_valid() {
if state.cursor_pos.0 != x || state.cursor_pos.1 != y {
let mut msg_out = Message::new();
msg_out.set_cursor_position(CursorPosition {
x,
y,
..Default::default()
});
let exclude = {
let now = get_time();
let lock = LATEST_INPUT_CURSOR.lock().unwrap();
if now - lock.time < 300 {
lock.conn
} else {
0
}
};
sp.send_without(msg_out, exclude);
}
2021-03-29 15:59:14 +08:00
}
state.cursor_pos = (x, y);
2021-03-29 15:59:14 +08:00
}
sp.snapshot(|sps| {
let mut msg_out = Message::new();
msg_out.set_cursor_position(CursorPosition {
x: state.cursor_pos.0,
y: state.cursor_pos.1,
..Default::default()
});
sps.send(msg_out);
Ok(())
})?;
Ok(())
}
fn run_cursor(sp: MouseCursorService, state: &mut StateCursor) -> ResultType<()> {
if let Some(hcursor) = crate::get_cursor()? {
if hcursor != state.hcursor {
let msg;
if let Some(cached) = state.cached_cursor_data.get(&hcursor) {
super::log::trace!("Cursor data cached, hcursor: {}", hcursor);
msg = cached.clone();
} else {
let mut data = crate::get_cursor_data(hcursor)?;
data.colors =
hbb_common::compress::compress(&data.colors[..], COMPRESS_LEVEL).into();
2021-03-29 15:59:14 +08:00
let mut tmp = Message::new();
tmp.set_cursor_data(data);
msg = Arc::new(tmp);
state.cached_cursor_data.insert(hcursor, msg.clone());
super::log::trace!("Cursor data updated, hcursor: {}", hcursor);
}
state.hcursor = hcursor;
sp.send_shared(msg.clone());
state.cursor_data = msg;
}
}
sp.snapshot(|sps| {
sps.send_shared(state.cursor_data.clone());
Ok(())
})?;
Ok(())
}
lazy_static::lazy_static! {
2022-03-08 15:42:58 +08:00
static ref ENIGO: Arc<Mutex<Enigo>> = {
Arc::new(Mutex::new(Enigo::new()))
};
2022-01-14 01:15:03 +08:00
static ref KEYS_DOWN: Arc<Mutex<HashMap<u64, Instant>>> = Default::default();
static ref LATEST_INPUT_CURSOR: Arc<Mutex<Input>> = Default::default();
static ref LATEST_CURSOR_POS: Arc<Mutex<(Instant, (i32, i32))>> = Arc::new(Mutex::new((Instant::now().sub(MOUSE_MOVE_PROTECTION_TIMEOUT), (0, 0))));
2021-03-29 15:59:14 +08:00
}
2021-08-12 01:25:32 +08:00
static EXITING: AtomicBool = AtomicBool::new(false);
2021-03-29 15:59:14 +08:00
const MOUSE_MOVE_PROTECTION_TIMEOUT: Duration = Duration::from_millis(1_000);
const MOUSE_ACTIVE_DISTANCE: i32 = 5;
2021-03-29 15:59:14 +08:00
// mac key input must be run in main thread, otherwise crash on >= osx 10.15
#[cfg(target_os = "macos")]
lazy_static::lazy_static! {
static ref QUEUE: Queue = Queue::main();
static ref IS_SERVER: bool = std::env::args().nth(1) == Some("--server".to_owned());
}
#[cfg(target_os = "linux")]
pub async fn set_uinput() -> ResultType<()> {
// Keyboard and mouse both open /dev/uinput
// TODO: Make sure there's no race
let keyboard = super::uinput::client::UInputKeyboard::new().await?;
log::info!("UInput keyboard created");
let mouse = super::uinput::client::UInputMouse::new().await?;
log::info!("UInput mouse created");
let xxx = ENIGO.lock();
let mut en = xxx.unwrap();
en.set_uinput_keyboard(Some(Box::new(keyboard)));
en.set_uinput_mouse(Some(Box::new(mouse)));
Ok(())
}
#[cfg(target_os = "linux")]
pub async fn set_uinput_resolution(minx: i32, maxx: i32, miny: i32, maxy: i32) -> ResultType<()> {
super::uinput::client::set_resolution(minx, maxx, miny, maxy).await
}
2021-03-29 15:59:14 +08:00
pub fn is_left_up(evt: &MouseEvent) -> bool {
let buttons = evt.mask >> 3;
let evt_type = evt.mask & 0x7;
return buttons == 1 && evt_type == 2;
}
#[cfg(windows)]
pub fn mouse_move_relative(x: i32, y: i32) {
crate::platform::windows::try_change_desktop();
let mut en = ENIGO.lock().unwrap();
en.mouse_move_relative(x, y);
}
#[cfg(not(target_os = "macos"))]
fn modifier_sleep() {
// sleep for a while, this is only for keying in rdp in peer so far
#[cfg(windows)]
std::thread::sleep(std::time::Duration::from_nanos(1));
}
2021-06-01 13:05:04 +08:00
#[inline]
fn get_modifier_state(key: Key, en: &mut Enigo) -> bool {
2022-03-07 20:16:28 +08:00
// https://github.com/rustdesk/rustdesk/issues/332
// on Linux, if RightAlt is down, RightAlt status is false, Alt status is true
// but on Windows, both are true
2021-06-01 13:05:04 +08:00
let x = en.get_key_state(key.clone());
match key {
Key::Shift => x || en.get_key_state(Key::RightShift),
Key::Control => x || en.get_key_state(Key::RightControl),
Key::Alt => x || en.get_key_state(Key::RightAlt),
Key::Meta => x || en.get_key_state(Key::RWin),
Key::RightShift => x || en.get_key_state(Key::Shift),
Key::RightControl => x || en.get_key_state(Key::Control),
Key::RightAlt => x || en.get_key_state(Key::Alt),
Key::RWin => x || en.get_key_state(Key::Meta),
2021-06-01 13:05:04 +08:00
_ => x,
}
}
2021-03-29 15:59:14 +08:00
pub fn handle_mouse(evt: &MouseEvent, conn: i32) {
2021-05-20 17:19:37 +08:00
#[cfg(target_os = "macos")]
if !*IS_SERVER {
// having GUI, run main GUI thread, otherwise crash
let evt = evt.clone();
QUEUE.exec_async(move || handle_mouse_(&evt, conn));
return;
}
handle_mouse_(evt, conn);
}
2021-08-09 04:21:42 +08:00
pub fn fix_key_down_timeout_loop() {
std::thread::spawn(move || loop {
2022-03-07 16:19:10 +08:00
std::thread::sleep(std::time::Duration::from_millis(1_000));
2021-08-09 04:21:42 +08:00
fix_key_down_timeout(false);
});
2021-08-12 01:25:32 +08:00
if let Err(err) = ctrlc::set_handler(move || {
fix_key_down_timeout_at_exit();
std::process::exit(0); // will call atexit on posix, but not on Windows
}) {
log::error!("Failed to set Ctrl-C handler: {}", err);
2021-08-09 04:21:42 +08:00
}
}
2021-08-12 01:25:32 +08:00
pub fn fix_key_down_timeout_at_exit() {
if EXITING.load(Ordering::SeqCst) {
2021-08-09 04:21:42 +08:00
return;
}
2021-08-12 01:25:32 +08:00
EXITING.store(true, Ordering::SeqCst);
2021-08-09 04:21:42 +08:00
fix_key_down_timeout(true);
log::info!("fix_key_down_timeout_at_exit");
}
2022-01-14 01:15:03 +08:00
#[inline]
fn get_layout(key: u32) -> Key {
Key::Layout(std::char::from_u32(key).unwrap_or('\0'))
}
2021-08-09 04:21:42 +08:00
fn fix_key_down_timeout(force: bool) {
if KEYS_DOWN.lock().unwrap().is_empty() {
return;
}
let cloned = (*KEYS_DOWN.lock().unwrap()).clone();
for (key, value) in cloned.into_iter() {
2022-03-05 15:16:10 +08:00
if force || value.elapsed().as_millis() >= 360_000 {
2021-08-09 04:21:42 +08:00
KEYS_DOWN.lock().unwrap().remove(&key);
let key = if key < KEY_CHAR_START {
2022-01-14 01:15:03 +08:00
if let Some(key) = KEY_MAP.get(&(key as _)) {
2021-08-09 04:21:42 +08:00
Some(*key)
} else {
2021-08-09 04:21:42 +08:00
None
}
2022-01-07 18:03:55 +08:00
} else {
2022-01-14 01:15:03 +08:00
Some(get_layout((key - KEY_CHAR_START) as _))
2021-08-09 04:21:42 +08:00
};
if let Some(key) = key {
let func = move || {
let mut en = ENIGO.lock().unwrap();
2022-03-07 16:19:10 +08:00
en.key_up(key);
log::debug!("Fixed {:?} timeout", key);
2021-08-09 04:21:42 +08:00
};
#[cfg(target_os = "macos")]
QUEUE.exec_async(func);
#[cfg(not(target_os = "macos"))]
func();
}
}
2021-08-09 04:21:42 +08:00
}
}
// e.g. current state of ctrl is down, but ctrl not in modifier, we should change ctrl to up, to make modifier state sync between remote and local
#[inline]
fn fix_modifier(
modifiers: &[EnumOrUnknown<ControlKey>],
key0: ControlKey,
key1: Key,
en: &mut Enigo,
) {
if get_modifier_state(key1, en) && !modifiers.contains(&EnumOrUnknown::new(key0)) {
2022-03-07 20:16:28 +08:00
#[cfg(windows)]
if key0 == ControlKey::Control && get_modifier_state(Key::Alt, en) {
// AltGr case
return;
}
en.key_up(key1);
log::debug!("Fixed {:?}", key1);
}
}
fn fix_modifiers(modifiers: &[EnumOrUnknown<ControlKey>], en: &mut Enigo, ck: i32) {
if ck != ControlKey::Shift.value() {
fix_modifier(modifiers, ControlKey::Shift, Key::Shift, en);
}
if ck != ControlKey::RShift.value() {
fix_modifier(modifiers, ControlKey::Shift, Key::RightShift, en);
}
if ck != ControlKey::Alt.value() {
fix_modifier(modifiers, ControlKey::Alt, Key::Alt, en);
}
if ck != ControlKey::RAlt.value() {
fix_modifier(modifiers, ControlKey::Alt, Key::RightAlt, en);
}
if ck != ControlKey::Control.value() {
fix_modifier(modifiers, ControlKey::Control, Key::Control, en);
}
if ck != ControlKey::RControl.value() {
fix_modifier(modifiers, ControlKey::Control, Key::RightControl, en);
}
if ck != ControlKey::Meta.value() {
fix_modifier(modifiers, ControlKey::Meta, Key::Meta, en);
}
if ck != ControlKey::RWin.value() {
fix_modifier(modifiers, ControlKey::Meta, Key::RWin, en);
}
}
fn active_mouse_(conn: i32) -> bool {
// out of time protection
if LATEST_CURSOR_POS.lock().unwrap().0.elapsed() > MOUSE_MOVE_PROTECTION_TIMEOUT {
return true;
}
let mut last_input = LATEST_INPUT_CURSOR.lock().unwrap();
// last conn input may be protected
if last_input.conn != conn {
return false;
}
// check if input is in valid range
match crate::get_cursor_pos() {
Some((x, y)) => {
let can_active = (last_input.x - x).abs() < MOUSE_ACTIVE_DISTANCE
&& (last_input.y - y).abs() < MOUSE_ACTIVE_DISTANCE;
if !can_active {
last_input.x = -MOUSE_ACTIVE_DISTANCE * 2;
last_input.y = -MOUSE_ACTIVE_DISTANCE * 2;
}
can_active
}
None => true,
}
}
2021-05-20 17:19:37 +08:00
fn handle_mouse_(evt: &MouseEvent, conn: i32) {
2021-08-12 01:25:32 +08:00
if EXITING.load(Ordering::SeqCst) {
2021-08-09 04:21:42 +08:00
return;
}
if !active_mouse_(conn) {
return;
}
2021-03-29 15:59:14 +08:00
#[cfg(windows)]
crate::platform::windows::try_change_desktop();
let buttons = evt.mask >> 3;
let evt_type = evt.mask & 0x7;
if evt_type == 0 {
let time = get_time();
*LATEST_INPUT_CURSOR.lock().unwrap() = Input {
time,
conn,
x: evt.x,
y: evt.y,
};
2021-03-29 15:59:14 +08:00
}
let mut en = ENIGO.lock().unwrap();
#[cfg(not(target_os = "macos"))]
let mut to_release = Vec::new();
if evt_type == 1 {
2022-03-08 12:08:18 +08:00
fix_modifiers(&evt.modifiers[..], &mut en, 0);
#[cfg(target_os = "macos")]
en.reset_flag();
for ref ck in evt.modifiers.iter() {
if let Some(key) = KEY_MAP.get(&ck.value()) {
2021-03-29 15:59:14 +08:00
#[cfg(target_os = "macos")]
en.add_flag(key);
#[cfg(not(target_os = "macos"))]
if key != &Key::CapsLock && key != &Key::NumLock {
2021-06-01 13:05:04 +08:00
if !get_modifier_state(key.clone(), &mut en) {
2021-03-29 15:59:14 +08:00
en.key_down(key.clone()).ok();
modifier_sleep();
to_release.push(key);
}
}
}
}
}
match evt_type {
0 => {
en.mouse_move_to(evt.x, evt.y);
}
1 => match buttons {
1 => {
allow_err!(en.mouse_down(MouseButton::Left));
}
2 => {
allow_err!(en.mouse_down(MouseButton::Right));
}
4 => {
allow_err!(en.mouse_down(MouseButton::Middle));
}
_ => {}
},
2 => match buttons {
1 => {
en.mouse_up(MouseButton::Left);
}
2 => {
en.mouse_up(MouseButton::Right);
}
4 => {
en.mouse_up(MouseButton::Middle);
}
_ => {}
},
3 => {
#[allow(unused_mut)]
let mut x = evt.x;
#[allow(unused_mut)]
let mut y = evt.y;
#[cfg(not(windows))]
{
x = -x;
y = -y;
}
// fix shift + scroll(down/up)
#[cfg(target_os = "macos")]
if evt
.modifiers
.contains(&EnumOrUnknown::new(ControlKey::Shift))
{
x = y;
y = 0;
}
2021-03-29 15:59:14 +08:00
if x != 0 {
en.mouse_scroll_x(x);
}
if y != 0 {
en.mouse_scroll_y(y);
}
}
_ => {}
}
#[cfg(not(target_os = "macos"))]
for key in to_release {
en.key_up(key.clone());
}
}
pub fn is_enter(evt: &KeyEvent) -> bool {
if let Some(key_event::Union::ControlKey(ck)) = evt.union {
2021-03-29 15:59:14 +08:00
if ck.value() == ControlKey::Return.value() || ck.value() == ControlKey::NumpadEnter.value()
{
return true;
}
}
return false;
}
pub async fn lock_screen() {
2022-04-24 02:37:27 +08:00
cfg_if::cfg_if! {
if #[cfg(target_os = "linux")] {
// xdg_screensaver lock not work on Linux from our service somehow
// loginctl lock-session also not work, they both work run rustdesk from cmd
std::thread::spawn(|| {
let mut key_event = KeyEvent::new();
2022-04-24 02:37:27 +08:00
key_event.set_chr('l' as _);
key_event.modifiers.push(ControlKey::Meta.into());
2022-07-19 16:04:23 +08:00
key_event.mode = KeyboardMode::Legacy.into();
key_event.down = true;
2022-04-24 02:37:27 +08:00
handle_key(&key_event);
2022-07-19 16:04:23 +08:00
2022-04-24 02:37:27 +08:00
key_event.down = false;
handle_key(&key_event);
});
} else if #[cfg(target_os = "macos")] {
// CGSession -suspend not real lock screen, it is user switch
std::thread::spawn(|| {
let mut key_event = KeyEvent::new();
2022-04-24 02:37:27 +08:00
key_event.set_chr('q' as _);
key_event.modifiers.push(ControlKey::Meta.into());
key_event.modifiers.push(ControlKey::Control.into());
2022-07-19 16:04:23 +08:00
key_event.mode = KeyboardMode::Legacy.into();
key_event.down = true;
2022-04-24 02:37:27 +08:00
handle_key(&key_event);
key_event.down = false;
handle_key(&key_event);
});
} else {
crate::platform::lock_screen();
}
}
super::video_service::switch_to_primary().await;
2022-04-24 02:37:27 +08:00
}
2021-03-29 15:59:14 +08:00
lazy_static::lazy_static! {
static ref KEY_MAP: HashMap<i32, Key> =
2021-03-29 15:59:14 +08:00
[
(ControlKey::Alt, Key::Alt),
(ControlKey::Backspace, Key::Backspace),
(ControlKey::CapsLock, Key::CapsLock),
(ControlKey::Control, Key::Control),
(ControlKey::Delete, Key::Delete),
(ControlKey::DownArrow, Key::DownArrow),
(ControlKey::End, Key::End),
(ControlKey::Escape, Key::Escape),
(ControlKey::F1, Key::F1),
(ControlKey::F10, Key::F10),
(ControlKey::F11, Key::F11),
(ControlKey::F12, Key::F12),
(ControlKey::F2, Key::F2),
(ControlKey::F3, Key::F3),
(ControlKey::F4, Key::F4),
(ControlKey::F5, Key::F5),
(ControlKey::F6, Key::F6),
(ControlKey::F7, Key::F7),
(ControlKey::F8, Key::F8),
(ControlKey::F9, Key::F9),
(ControlKey::Home, Key::Home),
(ControlKey::LeftArrow, Key::LeftArrow),
(ControlKey::Meta, Key::Meta),
(ControlKey::Option, Key::Option),
(ControlKey::PageDown, Key::PageDown),
(ControlKey::PageUp, Key::PageUp),
(ControlKey::Return, Key::Return),
(ControlKey::RightArrow, Key::RightArrow),
(ControlKey::Shift, Key::Shift),
(ControlKey::Space, Key::Space),
(ControlKey::Tab, Key::Tab),
(ControlKey::UpArrow, Key::UpArrow),
(ControlKey::Numpad0, Key::Numpad0),
(ControlKey::Numpad1, Key::Numpad1),
(ControlKey::Numpad2, Key::Numpad2),
(ControlKey::Numpad3, Key::Numpad3),
(ControlKey::Numpad4, Key::Numpad4),
(ControlKey::Numpad5, Key::Numpad5),
(ControlKey::Numpad6, Key::Numpad6),
(ControlKey::Numpad7, Key::Numpad7),
(ControlKey::Numpad8, Key::Numpad8),
(ControlKey::Numpad9, Key::Numpad9),
(ControlKey::Cancel, Key::Cancel),
(ControlKey::Clear, Key::Clear),
(ControlKey::Menu, Key::Alt),
(ControlKey::Pause, Key::Pause),
(ControlKey::Kana, Key::Kana),
(ControlKey::Hangul, Key::Hangul),
(ControlKey::Junja, Key::Junja),
(ControlKey::Final, Key::Final),
(ControlKey::Hanja, Key::Hanja),
(ControlKey::Kanji, Key::Kanji),
(ControlKey::Convert, Key::Convert),
(ControlKey::Select, Key::Select),
(ControlKey::Print, Key::Print),
(ControlKey::Execute, Key::Execute),
(ControlKey::Snapshot, Key::Snapshot),
(ControlKey::Insert, Key::Insert),
(ControlKey::Help, Key::Help),
(ControlKey::Sleep, Key::Sleep),
(ControlKey::Separator, Key::Separator),
(ControlKey::Scroll, Key::Scroll),
(ControlKey::NumLock, Key::NumLock),
(ControlKey::RWin, Key::RWin),
(ControlKey::Apps, Key::Apps),
(ControlKey::Multiply, Key::Multiply),
(ControlKey::Add, Key::Add),
(ControlKey::Subtract, Key::Subtract),
(ControlKey::Decimal, Key::Decimal),
(ControlKey::Divide, Key::Divide),
(ControlKey::Equals, Key::Equals),
(ControlKey::NumpadEnter, Key::NumpadEnter),
(ControlKey::RAlt, Key::RightAlt),
(ControlKey::RControl, Key::RightControl),
(ControlKey::RShift, Key::RightShift),
2021-03-29 15:59:14 +08:00
].iter().map(|(a, b)| (a.value(), b.clone())).collect();
static ref NUMPAD_KEY_MAP: HashMap<i32, bool> =
[
(ControlKey::Home, true),
(ControlKey::UpArrow, true),
(ControlKey::PageUp, true),
(ControlKey::LeftArrow, true),
(ControlKey::RightArrow, true),
(ControlKey::End, true),
(ControlKey::DownArrow, true),
(ControlKey::PageDown, true),
(ControlKey::Insert, true),
(ControlKey::Delete, true),
].iter().map(|(a, b)| (a.value(), b.clone())).collect();
}
pub fn handle_key(evt: &KeyEvent) {
#[cfg(target_os = "macos")]
if !*IS_SERVER {
// having GUI, run main GUI thread, otherwise crash
let evt = evt.clone();
QUEUE.exec_async(move || handle_key_(&evt));
return;
}
handle_key_(evt);
}
2022-07-18 17:09:32 +08:00
fn rdev_key_down_or_up(key: RdevKey, down_or_up: bool) {
let event_type = match down_or_up {
true => EventType::KeyPress(key),
false => EventType::KeyRelease(key),
};
2022-07-18 16:54:54 +08:00
let delay = std::time::Duration::from_millis(20);
2022-07-18 17:09:32 +08:00
match simulate(&event_type) {
2022-07-18 16:54:54 +08:00
Ok(()) => (),
Err(_simulate_error) => {
2022-07-26 10:30:26 +08:00
log::error!("Could not send {:?}", &event_type);
2022-07-18 16:54:54 +08:00
}
}
// Let ths OS catchup (at least MacOS)
std::thread::sleep(delay);
}
fn rdev_key_click(key: RdevKey) {
rdev_key_down_or_up(key, true);
rdev_key_down_or_up(key, false);
}
fn sync_status(evt: &KeyEvent) -> (bool, bool) {
/* todo! Shift+delete */
2022-07-18 16:54:54 +08:00
let mut en = ENIGO.lock().unwrap();
2022-07-18 17:19:33 +08:00
// remote caps status
2022-07-18 16:54:54 +08:00
let caps_locking = evt
.modifiers
.iter()
.position(|&r| r == ControlKey::CapsLock.into())
.is_some();
2022-07-18 17:19:33 +08:00
// remote numpad status
let num_locking = evt
.modifiers
.iter()
.position(|&r| r == ControlKey::NumLock.into())
.is_some();
let click_capslock = (caps_locking && !en.get_key_state(enigo::Key::CapsLock))
|| (!caps_locking && en.get_key_state(enigo::Key::CapsLock));
let click_numlock = (num_locking && !en.get_key_state(enigo::Key::NumLock))
|| (!num_locking && en.get_key_state(enigo::Key::NumLock));
2022-10-28 19:43:44 +08:00
#[cfg(windows)]
2022-10-29 02:14:38 +08:00
let click_numlock = {
2022-10-28 19:43:44 +08:00
let code = evt.chr();
let key = rdev::get_win_key(code, 0);
match key {
RdevKey::Home
| RdevKey::UpArrow
| RdevKey::PageUp
| RdevKey::LeftArrow
| RdevKey::RightArrow
| RdevKey::End
| RdevKey::DownArrow
| RdevKey::PageDown
| RdevKey::Insert
| RdevKey::Delete => en.get_key_state(enigo::Key::NumLock),
2022-10-29 02:14:38 +08:00
_ => click_numlock,
2022-10-28 19:43:44 +08:00
}
};
2022-10-29 13:37:20 +08:00
return (click_capslock, click_numlock);
}
fn map_keyboard_mode(evt: &KeyEvent) {
// map mode(1): Send keycode according to the peer platform.
#[cfg(windows)]
crate::platform::windows::try_change_desktop();
let (click_capslock, click_numlock) = sync_status(evt);
// Wayland
2022-09-06 14:02:12 +08:00
#[cfg(target_os = "linux")]
if !*IS_X11.lock().unwrap() {
let mut en = ENIGO.lock().unwrap();
let code = evt.chr() as u16;
#[cfg(not(target_os = "macos"))]
if click_capslock {
en.key_click(enigo::Key::CapsLock);
}
#[cfg(not(target_os = "macos"))]
if click_numlock {
en.key_click(enigo::Key::NumLock);
}
#[cfg(target_os = "macos")]
en.key_down(enigo::Key::CapsLock);
if evt.down {
2022-09-05 19:52:38 +08:00
en.key_down(enigo::Key::Raw(code)).ok();
} else {
en.key_up(enigo::Key::Raw(code));
}
return;
}
2022-08-30 15:35:39 +08:00
#[cfg(not(target_os = "macos"))]
if click_capslock {
rdev_key_click(RdevKey::CapsLock);
}
2022-08-30 15:35:39 +08:00
#[cfg(not(target_os = "macos"))]
if click_numlock {
rdev_key_click(RdevKey::NumLock);
}
2022-08-30 15:35:39 +08:00
#[cfg(target_os = "macos")]
if evt.down && click_capslock {
rdev_key_down_or_up(RdevKey::CapsLock, evt.down);
}
2022-07-21 11:31:17 +08:00
rdev_key_down_or_up(RdevKey::Unknown(evt.chr()), evt.down);
return;
}
2022-07-26 10:30:26 +08:00
fn legacy_keyboard_mode(evt: &KeyEvent) {
let (click_capslock, click_numlock) = sync_status(evt);
2021-03-29 15:59:14 +08:00
#[cfg(windows)]
crate::platform::windows::try_change_desktop();
let mut en = ENIGO.lock().unwrap();
if click_capslock {
2022-09-05 23:50:42 +08:00
en.key_click(Key::CapsLock);
}
if click_numlock {
2022-09-05 23:50:42 +08:00
en.key_click(Key::NumLock);
}
2021-03-29 15:59:14 +08:00
// disable numlock if press home etc when numlock is on,
// because we will get numpad value (7,8,9 etc) if not
#[cfg(windows)]
let mut _disable_numlock = false;
2021-03-29 15:59:14 +08:00
#[cfg(target_os = "macos")]
en.reset_flag();
2022-08-31 18:54:31 +08:00
// When long-pressed the command key, then press and release
2022-08-13 08:12:45 +08:00
// the Tab key, there should be CGEventFlagCommand in the flag.
#[cfg(target_os = "macos")]
2022-08-31 18:54:31 +08:00
for ck in evt.modifiers.iter() {
if let Some(key) = KEY_MAP.get(&ck.value()) {
2022-08-13 08:12:45 +08:00
en.add_flag(key);
}
}
2021-03-29 15:59:14 +08:00
#[cfg(not(target_os = "macos"))]
let mut to_release = Vec::new();
2022-09-05 23:50:42 +08:00
if evt.down {
let ck = if let Some(key_event::Union::ControlKey(ck)) = evt.union {
ck.value()
} else {
-1
};
fix_modifiers(&evt.modifiers[..], &mut en, ck);
for ref ck in evt.modifiers.iter() {
if let Some(key) = KEY_MAP.get(&ck.value()) {
2022-03-08 12:08:18 +08:00
#[cfg(target_os = "linux")]
if key == &Key::Alt && !get_modifier_state(key.clone(), &mut en) {
// for AltGr on Linux
if KEYS_DOWN
.lock()
.unwrap()
.get(&(ControlKey::RAlt.value() as _))
.is_some()
{
continue;
}
}
#[cfg(not(target_os = "macos"))]
2022-09-04 14:29:14 +08:00
if !get_modifier_state(key.clone(), &mut en) {
2022-09-05 23:50:42 +08:00
en.key_down(key.clone()).ok();
2022-09-04 14:29:14 +08:00
modifier_sleep();
to_release.push(key);
2021-03-29 15:59:14 +08:00
}
}
}
}
2022-09-04 14:29:14 +08:00
2021-03-29 15:59:14 +08:00
match evt.union {
Some(key_event::Union::ControlKey(ck)) => {
2021-03-29 15:59:14 +08:00
if let Some(key) = KEY_MAP.get(&ck.value()) {
#[cfg(windows)]
if let Some(_) = NUMPAD_KEY_MAP.get(&ck.value()) {
_disable_numlock = en.get_key_state(Key::NumLock);
if _disable_numlock {
en.key_down(Key::NumLock).ok();
en.key_up(Key::NumLock);
2021-03-29 15:59:14 +08:00
}
}
if evt.down {
2022-09-05 23:50:42 +08:00
en.key_down(key.clone()).ok();
2022-01-14 01:15:03 +08:00
KEYS_DOWN
.lock()
.unwrap()
.insert(ck.value() as _, Instant::now());
2021-03-29 15:59:14 +08:00
} else {
2022-09-05 23:50:42 +08:00
en.key_up(key.clone());
2022-01-14 01:15:03 +08:00
KEYS_DOWN.lock().unwrap().remove(&(ck.value() as _));
2021-03-29 15:59:14 +08:00
}
} else if ck.value() == ControlKey::CtrlAltDel.value() {
// have to spawn new thread because send_sas is tokio_main, the caller can not be tokio_main.
std::thread::spawn(|| {
allow_err!(send_sas());
});
} else if ck.value() == ControlKey::LockScreen.value() {
lock_screen_2();
2021-03-29 15:59:14 +08:00
}
}
Some(key_event::Union::Chr(chr)) => {
2021-03-29 15:59:14 +08:00
if evt.down {
2022-09-05 23:50:42 +08:00
if en.key_down(get_layout(chr)).is_ok() {
KEYS_DOWN
.lock()
.unwrap()
.insert(chr as u64 + KEY_CHAR_START, Instant::now());
2022-09-03 03:29:25 +08:00
} else {
2022-09-05 23:50:42 +08:00
if let Ok(chr) = char::try_from(chr) {
let mut x = chr.to_string();
if get_modifier_state(Key::Shift, &mut en)
|| get_modifier_state(Key::CapsLock, &mut en)
{
x = x.to_uppercase();
2022-09-03 03:29:25 +08:00
}
2022-09-05 23:50:42 +08:00
en.key_sequence(&x);
2022-09-03 03:29:25 +08:00
}
}
2022-08-31 18:54:31 +08:00
KEYS_DOWN
.lock()
.unwrap()
.insert(chr as u64 + KEY_CHAR_START, Instant::now());
2022-09-03 03:29:25 +08:00
} else {
2022-09-05 23:50:42 +08:00
en.key_up(get_layout(chr));
KEYS_DOWN
.lock()
.unwrap()
2022-01-14 01:15:03 +08:00
.remove(&(chr as u64 + KEY_CHAR_START));
2021-03-29 15:59:14 +08:00
}
}
Some(key_event::Union::Unicode(chr)) => {
2021-03-29 15:59:14 +08:00
if let Ok(chr) = char::try_from(chr) {
en.key_sequence(&chr.to_string());
}
}
Some(key_event::Union::Seq(ref seq)) => {
2021-03-29 15:59:14 +08:00
en.key_sequence(&seq);
}
_ => {}
}
#[cfg(not(target_os = "macos"))]
for key in to_release {
2022-09-05 23:50:42 +08:00
en.key_up(key.clone());
2022-08-02 18:47:29 +08:00
}
2022-07-28 11:01:42 +08:00
}
2022-07-26 10:30:26 +08:00
fn handle_key_(evt: &KeyEvent) {
if EXITING.load(Ordering::SeqCst) {
return;
}
2022-07-19 15:43:13 +08:00
match evt.mode.unwrap() {
KeyboardMode::Legacy => {
2022-07-26 10:30:26 +08:00
legacy_keyboard_mode(evt);
2022-07-19 14:45:54 +08:00
}
2022-07-19 15:43:13 +08:00
KeyboardMode::Map => {
2022-07-26 10:30:26 +08:00
map_keyboard_mode(evt);
}
2022-07-28 11:01:42 +08:00
KeyboardMode::Translate => {
2022-09-05 23:50:42 +08:00
legacy_keyboard_mode(evt);
2022-07-28 11:01:42 +08:00
}
_ => {
2022-07-26 10:30:26 +08:00
legacy_keyboard_mode(evt);
}
}
}
#[tokio::main(flavor = "current_thread")]
async fn lock_screen_2() {
lock_screen().await;
}
2021-06-25 19:42:51 +08:00
#[tokio::main(flavor = "current_thread")]
2021-03-29 15:59:14 +08:00
async fn send_sas() -> ResultType<()> {
let mut stream = crate::ipc::connect(1000, crate::POSTFIX_SERVICE).await?;
timeout(1000, stream.send(&crate::ipc::Data::SAS)).await??;
Ok(())
}
#[cfg(test)]
mod test {
use super::*;
2022-07-18 23:04:04 +08:00
use rdev::{listen, Event, EventType, Key};
use std::sync::mpsc;
#[test]
fn test_handle_key() {
// listen
let (tx, rx) = mpsc::channel();
std::thread::spawn(move || {
std::env::set_var("KEYBOARD_ONLY", "y");
let func = move |event: Event| {
tx.send(event).ok();
};
if let Err(error) = listen(func) {
println!("Error: {:?}", error);
}
});
// set key/char base on char
let mut evt = KeyEvent::new();
2022-07-18 16:54:54 +08:00
evt.set_chr(66);
2022-07-19 16:15:57 +08:00
evt.mode = KeyboardMode::Legacy.into();
2022-07-18 16:54:54 +08:00
evt.modifiers.push(ControlKey::CapsLock.into());
// press
evt.down = true;
handle_key(&evt);
if let Ok(listen_evt) = rx.recv() {
assert_eq!(listen_evt.event_type, EventType::KeyPress(Key::Num1))
}
// release
evt.down = false;
handle_key(&evt);
if let Ok(listen_evt) = rx.recv() {
assert_eq!(listen_evt.event_type, EventType::KeyRelease(Key::Num1))
}
}
2022-07-18 23:04:04 +08:00
#[test]
fn test_get_key_state() {
let mut en = ENIGO.lock().unwrap();
println!(
"[*] test_get_key_state: {:?}",
en.get_key_state(enigo::Key::NumLock)
);
}
}