#![allow(improper_ctypes_definitions)]
use crate::eventhooks::macos::common::*;
use crate::eventhooks::types::{Event, GrabError};
use cocoa::base::nil;
use cocoa::foundation::NSAutoreleasePool;
use core_graphics::event::{CGEventTapLocation, CGEventType};
use std::os::raw::c_void;
use std::sync::Mutex;
type GrabCallbackBox = Box<dyn FnMut(Event) -> Option<Event> + Send>;
static GLOBAL_CALLBACK: Mutex<Option<GrabCallbackBox>> = Mutex::new(None);
static mut CUR_LOOP: CFRunLoopSourceRef = std::ptr::null_mut();
unsafe extern "C" fn raw_callback(
_proxy: CGEventTapProxy,
_type: CGEventType,
cg_event: CGEventRef,
_user_info: *mut c_void,
) -> CGEventRef {
if let Some(event) = unsafe { convert(_type, &cg_event) } {
if let Ok(mut guard) = GLOBAL_CALLBACK.lock() {
if let Some(callback) = guard.as_mut() {
if callback(event).is_none() {
cg_event.set_type(CGEventType::Null);
}
}
}
}
cg_event
}
#[inline]
pub fn is_grabbed() -> bool {
unsafe { !CUR_LOOP.is_null() }
}
pub fn grab<T>(callback: T) -> Result<(), GrabError>
where
T: FnMut(Event) -> Option<Event> + Send + 'static,
{
if is_grabbed() {
return Ok(());
}
if let Ok(mut guard) = GLOBAL_CALLBACK.lock() {
*guard = Some(Box::new(callback));
}
unsafe {
let _pool = NSAutoreleasePool::new(nil);
let tap = CGEventTapCreate(
CGEventTapLocation::Session,
kCGHeadInsertEventTap,
CGEventTapOption::Default,
kCGEventMaskForAllEvents,
raw_callback,
nil,
);
if tap.is_null() {
return Err(GrabError::EventTapError);
}
let loop_source = CFMachPortCreateRunLoopSource(nil, tap, 0);
if loop_source.is_null() {
return Err(GrabError::LoopSourceError);
}
CUR_LOOP = CFRunLoopGetCurrent() as _;
CFRunLoopAddSource(CUR_LOOP, loop_source, kCFRunLoopCommonModes);
CGEventTapEnable(tap, true);
CFRunLoopRun();
}
Ok(())
}
pub fn exit_grab() -> Result<(), GrabError> {
unsafe {
if !CUR_LOOP.is_null() {
CFRunLoopStop(CUR_LOOP);
CUR_LOOP = std::ptr::null_mut();
}
}
Ok(())
}