#include "bridge.h" #import #include "AppDelegate.h" #include extern "C" { } #include void register_keypress_callback(void * self, KeypressCallback callback) { keypress_callback = callback; interceptor_instance = self; } int32_t initialize() { AppDelegate *delegate = [[AppDelegate alloc] init]; NSApplication * application = [NSApplication sharedApplication]; [application setDelegate:delegate]; } int32_t eventloop() { [NSApp run]; } void send_string(const char * string) { char * stringCopy = strdup(string); dispatch_async(dispatch_get_main_queue(), ^(void) { // Convert the c string to a UniChar array as required by the CGEventKeyboardSetUnicodeString method NSString *nsString = [NSString stringWithUTF8String:stringCopy];CFStringRef cfString = (__bridge CFStringRef) nsString; std::vector buffer(nsString.length); CFStringGetCharacters(cfString, CFRangeMake(0, nsString.length), buffer.data()); free(stringCopy); // Send the event // Because of a bug ( or undocumented limit ) of the CGEventKeyboardSetUnicodeString method // the string gets truncated after 20 characters, so we need to send multiple events. int i = 0; while (i < buffer.size()) { int chunk_size = 20; if ((i+chunk_size) > buffer.size()) { chunk_size = buffer.size() - i; } UniChar * offset_buffer = buffer.data() + i; CGEventRef e = CGEventCreateKeyboardEvent(NULL, 0x31, true); CGEventKeyboardSetUnicodeString(e, chunk_size, offset_buffer); CGEventPost(kCGHIDEventTap, e); CFRelease(e); usleep(2000); i += chunk_size; } }); } void delete_string(int32_t count) { dispatch_async(dispatch_get_main_queue(), ^(void) { for (int i = 0; i < count; i++) { CGEventRef keydown; keydown = CGEventCreateKeyboardEvent(NULL, 0x33, true); CGEventPost(kCGHIDEventTap, keydown); CFRelease(keydown); usleep(2000); CGEventRef keyup; keyup = CGEventCreateKeyboardEvent(NULL, 0x33, false); CGEventPost(kCGHIDEventTap, keyup); CFRelease(keyup); usleep(2000); } }); } void send_vkey(int32_t vk) { dispatch_async(dispatch_get_main_queue(), ^(void) { CGEventRef keydown; keydown = CGEventCreateKeyboardEvent(NULL, vk, true); CGEventPost(kCGHIDEventTap, keydown); CFRelease(keydown); usleep(2000); CGEventRef keyup; keyup = CGEventCreateKeyboardEvent(NULL, vk, false); CGEventPost(kCGHIDEventTap, keyup); CFRelease(keyup); usleep(2000); }); } int32_t get_active_app_bundle(char * buffer, int32_t size) { NSRunningApplication *frontApp = [[NSWorkspace sharedWorkspace] frontmostApplication]; NSString *bundlePath = [frontApp bundleURL].path; const char * path = [bundlePath UTF8String]; snprintf(buffer, size, "%s", path); return 1; } int32_t get_active_app_identifier(char * buffer, int32_t size) { NSRunningApplication *frontApp = [[NSWorkspace sharedWorkspace] frontmostApplication]; NSString *bundlePath = frontApp.bundleIdentifier; const char * path = [bundlePath UTF8String]; snprintf(buffer, size, "%s", path); return 1; }