/* * This file is part of espanso. * * Copyright (C) 2019-2021 Federico Terzi * * espanso 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 * (at your option) any later version. * * espanso 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 espanso. If not, see . */ #include "native.h" #include #import #import #include // Events dispatched by espanso are "marked" with a custom location // so that we can later skip them in the detect module. CGPoint ESPANSO_POINT_MARKER = CGPointMake(-27469, 0); void inject_string(char *string, int32_t delay) { long udelay = delay * 1000; 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 // Check if the shift key is down, and if so, release it // To see why: https://github.com/federico-terzi/espanso/issues/279 if (CGEventSourceKeyState(kCGEventSourceStateHIDSystemState, 0x38)) { CGEventRef e2 = CGEventCreateKeyboardEvent(NULL, 0x38, false); CGEventSetLocation(e2, ESPANSO_POINT_MARKER); CGEventPost(kCGHIDEventTap, e2); CFRelease(e2); usleep(udelay); } // 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); CGEventSetLocation(e, ESPANSO_POINT_MARKER); CGEventKeyboardSetUnicodeString(e, chunk_size, offset_buffer); CGEventPost(kCGHIDEventTap, e); CFRelease(e); usleep(udelay); // Some applications require an explicit release of the space key // For more information: https://github.com/federico-terzi/espanso/issues/159 CGEventRef e2 = CGEventCreateKeyboardEvent(NULL, 0x31, false); CGEventSetLocation(e2, ESPANSO_POINT_MARKER); CGEventPost(kCGHIDEventTap, e2); CFRelease(e2); usleep(udelay); i += chunk_size; } }); } void inject_separate_vkeys(int32_t *_vkey_array, int32_t vkey_count, int32_t delay) { long udelay = delay * 1000; // Create an heap allocated copy of the array, so that it doesn't get freed within the block int32_t *vkey_array = (int32_t*)malloc(sizeof(int32_t)*vkey_count); memcpy(vkey_array, _vkey_array, sizeof(int32_t)*vkey_count); dispatch_async(dispatch_get_main_queue(), ^(void) { for (int i = 0; i= 0; i--) { CGEventRef keyup; keyup = CGEventCreateKeyboardEvent(NULL, vkey_array[i], false); CGEventSetLocation(keyup, ESPANSO_POINT_MARKER); CGEventPost(kCGHIDEventTap, keyup); CFRelease(keyup); usleep(udelay); } free(vkey_array); }); }