obs-cocoa.c 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337
  1. /******************************************************************************
  2. Copyright (C) 2013 by Ruwen Hahn <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #include "util/platform.h"
  15. #include "util/dstr.h"
  16. #include "obs.h"
  17. #include "obs-internal.h"
  18. #include <unistd.h>
  19. #include <sys/types.h>
  20. #include <sys/sysctl.h>
  21. #include <objc/objc.h>
  22. #include <Carbon/Carbon.h>
  23. #include <IOKit/hid/IOHIDDevice.h>
  24. #include <IOKit/hid/IOHIDManager.h>
  25. const char *get_module_extension(void)
  26. {
  27. return ".so";
  28. }
  29. static const char *module_bin[] = {
  30. "../obs-plugins",
  31. OBS_INSTALL_PREFIX "obs-plugins",
  32. };
  33. static const char *module_data[] = {
  34. "../data/obs-plugins/%module%",
  35. OBS_INSTALL_DATA_PATH "obs-plugins/%module%",
  36. };
  37. static const int module_patterns_size =
  38. sizeof(module_bin)/sizeof(module_bin[0]);
  39. void add_default_module_paths(void)
  40. {
  41. for (int i = 0; i < module_patterns_size; i++)
  42. obs_add_module_path(module_bin[i], module_data[i]);
  43. }
  44. char *find_libobs_data_file(const char *file)
  45. {
  46. struct dstr path;
  47. dstr_init_copy(&path, OBS_INSTALL_DATA_PATH "/libobs/");
  48. dstr_cat(&path, file);
  49. return path.array;
  50. }
  51. static void log_processor_name(void)
  52. {
  53. char *name = NULL;
  54. size_t size;
  55. int ret;
  56. ret = sysctlbyname("machdep.cpu.brand_string", NULL, &size, NULL, 0);
  57. if (ret != 0)
  58. return;
  59. name = malloc(size);
  60. ret = sysctlbyname("machdep.cpu.brand_string", name, &size, NULL, 0);
  61. if (ret == 0)
  62. blog(LOG_INFO, "CPU Name: %s", name);
  63. free(name);
  64. }
  65. static void log_processor_speed(void)
  66. {
  67. size_t size;
  68. long long freq;
  69. int ret;
  70. size = sizeof(freq);
  71. ret = sysctlbyname("hw.cpufrequency", &freq, &size, NULL, 0);
  72. if (ret == 0)
  73. blog(LOG_INFO, "CPU Speed: %lldMHz", freq / 1000000);
  74. }
  75. static void log_processor_cores(void)
  76. {
  77. size_t size;
  78. int physical_cores = 0, logical_cores = 0;
  79. int ret;
  80. size = sizeof(physical_cores);
  81. ret = sysctlbyname("machdep.cpu.core_count", &physical_cores,
  82. &size, NULL, 0);
  83. if (ret != 0)
  84. return;
  85. ret = sysctlbyname("machdep.cpu.thread_count", &logical_cores,
  86. &size, NULL, 0);
  87. if (ret != 0)
  88. return;
  89. blog(LOG_INFO, "Physical Cores: %d, Logical Cores: %d",
  90. physical_cores, logical_cores);
  91. }
  92. static void log_available_memory(void)
  93. {
  94. size_t size;
  95. long long memory_available;
  96. int ret;
  97. size = sizeof(memory_available);
  98. ret = sysctlbyname("hw.memsize", &memory_available, &size, NULL, 0);
  99. if (ret == 0)
  100. blog(LOG_INFO, "Physical Memory: %lldMB Total",
  101. memory_available / 1024 / 1024);
  102. }
  103. static void log_os_name(id pi, SEL UTF8String)
  104. {
  105. unsigned long os_id = (unsigned long)objc_msgSend(pi,
  106. sel_registerName("operatingSystem"));
  107. id os = objc_msgSend(pi,
  108. sel_registerName("operatingSystemName"));
  109. const char *name = (const char*)objc_msgSend(os, UTF8String);
  110. if (os_id == 5 /*NSMACHOperatingSystem*/) {
  111. blog(LOG_INFO, "OS Name: Mac OS X (%s)", name);
  112. return;
  113. }
  114. blog(LOG_INFO, "OS Name: %s", name ? name : "Unknown");
  115. }
  116. static void log_os_version(id pi, SEL UTF8String)
  117. {
  118. id vs = objc_msgSend(pi,
  119. sel_registerName("operatingSystemVersionString"));
  120. const char *version = (const char*)objc_msgSend(vs, UTF8String);
  121. blog(LOG_INFO, "OS Version: %s", version ? version : "Unknown");
  122. }
  123. static void log_os(void)
  124. {
  125. Class NSProcessInfo = objc_getClass("NSProcessInfo");
  126. id pi = objc_msgSend((id)NSProcessInfo,
  127. sel_registerName("processInfo"));
  128. SEL UTF8String = sel_registerName("UTF8String");
  129. log_os_name(pi, UTF8String);
  130. log_os_version(pi, UTF8String);
  131. }
  132. static void log_kernel_version(void)
  133. {
  134. char kernel_version[1024];
  135. size_t size = sizeof(kernel_version);
  136. int ret;
  137. ret = sysctlbyname("kern.osrelease", kernel_version, &size,
  138. NULL, 0);
  139. if (ret == 0)
  140. blog(LOG_INFO, "Kernel Version: %s", kernel_version);
  141. }
  142. void log_system_info(void)
  143. {
  144. log_processor_name();
  145. log_processor_speed();
  146. log_processor_cores();
  147. log_available_memory();
  148. log_os();
  149. log_kernel_version();
  150. }
  151. static bool dstr_from_cfstring(struct dstr *str, CFStringRef ref)
  152. {
  153. CFIndex length = CFStringGetLength(ref);
  154. CFIndex max_size = CFStringGetMaximumSizeForEncoding(length,
  155. kCFStringEncodingUTF8);
  156. dstr_reserve(str, max_size);
  157. if (!CFStringGetCString(ref, str->array, max_size,
  158. kCFStringEncodingUTF8))
  159. return false;
  160. str->len = strlen(str->array);
  161. return true;
  162. }
  163. struct obs_hotkeys_platform {
  164. volatile long refs;
  165. TISInputSourceRef tis;
  166. CFDataRef layout_data;
  167. UCKeyboardLayout *layout;
  168. IOHIDManagerRef manager;
  169. DARRAY(IOHIDElementRef) keys[OBS_KEY_LAST_VALUE];
  170. };
  171. static void hotkeys_retain(struct obs_hotkeys_platform *plat)
  172. {
  173. os_atomic_inc_long(&plat->refs);
  174. }
  175. static inline void free_hotkeys_platform(obs_hotkeys_platform_t *plat);
  176. static void hotkeys_release(struct obs_hotkeys_platform *plat)
  177. {
  178. if (os_atomic_dec_long(&plat->refs) == -1)
  179. free_hotkeys_platform(plat);
  180. }
  181. #define INVALID_KEY 0xff
  182. int obs_key_to_virtual_key(obs_key_t code)
  183. {
  184. switch (code) {
  185. case OBS_KEY_A: return kVK_ANSI_A;
  186. case OBS_KEY_B: return kVK_ANSI_B;
  187. case OBS_KEY_C: return kVK_ANSI_C;
  188. case OBS_KEY_D: return kVK_ANSI_D;
  189. case OBS_KEY_E: return kVK_ANSI_E;
  190. case OBS_KEY_F: return kVK_ANSI_F;
  191. case OBS_KEY_G: return kVK_ANSI_G;
  192. case OBS_KEY_H: return kVK_ANSI_H;
  193. case OBS_KEY_I: return kVK_ANSI_I;
  194. case OBS_KEY_J: return kVK_ANSI_J;
  195. case OBS_KEY_K: return kVK_ANSI_K;
  196. case OBS_KEY_L: return kVK_ANSI_L;
  197. case OBS_KEY_M: return kVK_ANSI_M;
  198. case OBS_KEY_N: return kVK_ANSI_N;
  199. case OBS_KEY_O: return kVK_ANSI_O;
  200. case OBS_KEY_P: return kVK_ANSI_P;
  201. case OBS_KEY_Q: return kVK_ANSI_Q;
  202. case OBS_KEY_R: return kVK_ANSI_R;
  203. case OBS_KEY_S: return kVK_ANSI_S;
  204. case OBS_KEY_T: return kVK_ANSI_T;
  205. case OBS_KEY_U: return kVK_ANSI_U;
  206. case OBS_KEY_V: return kVK_ANSI_V;
  207. case OBS_KEY_W: return kVK_ANSI_W;
  208. case OBS_KEY_X: return kVK_ANSI_X;
  209. case OBS_KEY_Y: return kVK_ANSI_Y;
  210. case OBS_KEY_Z: return kVK_ANSI_Z;
  211. case OBS_KEY_1: return kVK_ANSI_1;
  212. case OBS_KEY_2: return kVK_ANSI_2;
  213. case OBS_KEY_3: return kVK_ANSI_3;
  214. case OBS_KEY_4: return kVK_ANSI_4;
  215. case OBS_KEY_5: return kVK_ANSI_5;
  216. case OBS_KEY_6: return kVK_ANSI_6;
  217. case OBS_KEY_7: return kVK_ANSI_7;
  218. case OBS_KEY_8: return kVK_ANSI_8;
  219. case OBS_KEY_9: return kVK_ANSI_9;
  220. case OBS_KEY_0: return kVK_ANSI_0;
  221. case OBS_KEY_RETURN: return kVK_Return;
  222. case OBS_KEY_ESCAPE: return kVK_Escape;
  223. case OBS_KEY_BACKSPACE: return kVK_Delete;
  224. case OBS_KEY_TAB: return kVK_Tab;
  225. case OBS_KEY_SPACE: return kVK_Space;
  226. case OBS_KEY_MINUS: return kVK_ANSI_Minus;
  227. case OBS_KEY_EQUAL: return kVK_ANSI_Equal;
  228. case OBS_KEY_BRACKETLEFT: return kVK_ANSI_LeftBracket;
  229. case OBS_KEY_BRACKETRIGHT: return kVK_ANSI_RightBracket;
  230. case OBS_KEY_BACKSLASH: return kVK_ANSI_Backslash;
  231. case OBS_KEY_SEMICOLON: return kVK_ANSI_Semicolon;
  232. case OBS_KEY_QUOTE: return kVK_ANSI_Quote;
  233. case OBS_KEY_DEAD_GRAVE: return kVK_ANSI_Grave;
  234. case OBS_KEY_COMMA: return kVK_ANSI_Comma;
  235. case OBS_KEY_PERIOD: return kVK_ANSI_Period;
  236. case OBS_KEY_SLASH: return kVK_ANSI_Slash;
  237. case OBS_KEY_CAPSLOCK: return kVK_CapsLock;
  238. case OBS_KEY_SECTION: return kVK_ISO_Section;
  239. case OBS_KEY_F1: return kVK_F1;
  240. case OBS_KEY_F2: return kVK_F2;
  241. case OBS_KEY_F3: return kVK_F3;
  242. case OBS_KEY_F4: return kVK_F4;
  243. case OBS_KEY_F5: return kVK_F5;
  244. case OBS_KEY_F6: return kVK_F6;
  245. case OBS_KEY_F7: return kVK_F7;
  246. case OBS_KEY_F8: return kVK_F8;
  247. case OBS_KEY_F9: return kVK_F9;
  248. case OBS_KEY_F10: return kVK_F10;
  249. case OBS_KEY_F11: return kVK_F11;
  250. case OBS_KEY_F12: return kVK_F12;
  251. case OBS_KEY_HELP: return kVK_Help;
  252. case OBS_KEY_HOME: return kVK_Home;
  253. case OBS_KEY_PAGEUP: return kVK_PageUp;
  254. case OBS_KEY_DELETE: return kVK_ForwardDelete;
  255. case OBS_KEY_END: return kVK_End;
  256. case OBS_KEY_PAGEDOWN: return kVK_PageDown;
  257. case OBS_KEY_RIGHT: return kVK_RightArrow;
  258. case OBS_KEY_LEFT: return kVK_LeftArrow;
  259. case OBS_KEY_DOWN: return kVK_DownArrow;
  260. case OBS_KEY_UP: return kVK_UpArrow;
  261. case OBS_KEY_CLEAR: return kVK_ANSI_KeypadClear;
  262. case OBS_KEY_NUMSLASH: return kVK_ANSI_KeypadDivide;
  263. case OBS_KEY_NUMASTERISK: return kVK_ANSI_KeypadMultiply;
  264. case OBS_KEY_NUMMINUS: return kVK_ANSI_KeypadMinus;
  265. case OBS_KEY_NUMPLUS: return kVK_ANSI_KeypadPlus;
  266. case OBS_KEY_ENTER: return kVK_ANSI_KeypadEnter;
  267. case OBS_KEY_NUM1: return kVK_ANSI_Keypad1;
  268. case OBS_KEY_NUM2: return kVK_ANSI_Keypad2;
  269. case OBS_KEY_NUM3: return kVK_ANSI_Keypad3;
  270. case OBS_KEY_NUM4: return kVK_ANSI_Keypad4;
  271. case OBS_KEY_NUM5: return kVK_ANSI_Keypad5;
  272. case OBS_KEY_NUM6: return kVK_ANSI_Keypad6;
  273. case OBS_KEY_NUM7: return kVK_ANSI_Keypad7;
  274. case OBS_KEY_NUM8: return kVK_ANSI_Keypad8;
  275. case OBS_KEY_NUM9: return kVK_ANSI_Keypad9;
  276. case OBS_KEY_NUM0: return kVK_ANSI_Keypad0;
  277. case OBS_KEY_NUMPERIOD: return kVK_ANSI_KeypadDecimal;
  278. case OBS_KEY_NUMEQUAL: return kVK_ANSI_KeypadEquals;
  279. case OBS_KEY_F13: return kVK_F13;
  280. case OBS_KEY_F14: return kVK_F14;
  281. case OBS_KEY_F15: return kVK_F15;
  282. case OBS_KEY_F16: return kVK_F16;
  283. case OBS_KEY_F17: return kVK_F17;
  284. case OBS_KEY_F18: return kVK_F18;
  285. case OBS_KEY_F19: return kVK_F19;
  286. case OBS_KEY_F20: return kVK_F20;
  287. case OBS_KEY_CONTROL: return kVK_Control;
  288. case OBS_KEY_SHIFT: return kVK_Shift;
  289. case OBS_KEY_ALT: return kVK_Option;
  290. case OBS_KEY_META: return kVK_Command;
  291. //case OBS_KEY_CONTROL: return kVK_RightControl;
  292. //case OBS_KEY_SHIFT: return kVK_RightShift;
  293. //case OBS_KEY_ALT: return kVK_RightOption;
  294. //case OBS_KEY_META: return 0x36;
  295. case OBS_KEY_NONE:
  296. case OBS_KEY_LAST_VALUE:
  297. default:
  298. break;
  299. }
  300. return INVALID_KEY;
  301. }
  302. static bool localized_key_to_str(obs_key_t key, struct dstr *str)
  303. {
  304. #define MAP_KEY(k, s) case k: \
  305. dstr_copy(str, obs_get_hotkey_translation(k, s)); \
  306. return true
  307. #define MAP_BUTTON(i) case OBS_KEY_MOUSE ## i: \
  308. dstr_copy(str, obs_get_hotkey_translation(key, "Mouse " #i)); \
  309. return true
  310. switch (key) {
  311. MAP_KEY(OBS_KEY_SPACE, "Space");
  312. MAP_KEY(OBS_KEY_NUMEQUAL, "= (Keypad)");
  313. MAP_KEY(OBS_KEY_NUMASTERISK, "* (Keypad)");
  314. MAP_KEY(OBS_KEY_NUMPLUS, "+ (Keypad)");
  315. MAP_KEY(OBS_KEY_NUMMINUS, "- (Keypad)");
  316. MAP_KEY(OBS_KEY_NUMPERIOD, ". (Keypad)");
  317. MAP_KEY(OBS_KEY_NUMSLASH, "/ (Keypad)");
  318. MAP_KEY(OBS_KEY_NUM0, "0 (Keypad)");
  319. MAP_KEY(OBS_KEY_NUM1, "1 (Keypad)");
  320. MAP_KEY(OBS_KEY_NUM2, "2 (Keypad)");
  321. MAP_KEY(OBS_KEY_NUM3, "3 (Keypad)");
  322. MAP_KEY(OBS_KEY_NUM4, "4 (Keypad)");
  323. MAP_KEY(OBS_KEY_NUM5, "5 (Keypad)");
  324. MAP_KEY(OBS_KEY_NUM6, "6 (Keypad)");
  325. MAP_KEY(OBS_KEY_NUM7, "7 (Keypad)");
  326. MAP_KEY(OBS_KEY_NUM8, "8 (Keypad)");
  327. MAP_KEY(OBS_KEY_NUM9, "9 (Keypad)");
  328. MAP_BUTTON(1);
  329. MAP_BUTTON(2);
  330. MAP_BUTTON(3);
  331. MAP_BUTTON(4);
  332. MAP_BUTTON(5);
  333. MAP_BUTTON(6);
  334. MAP_BUTTON(7);
  335. MAP_BUTTON(8);
  336. MAP_BUTTON(9);
  337. MAP_BUTTON(10);
  338. MAP_BUTTON(11);
  339. MAP_BUTTON(12);
  340. MAP_BUTTON(13);
  341. MAP_BUTTON(14);
  342. MAP_BUTTON(15);
  343. MAP_BUTTON(16);
  344. MAP_BUTTON(17);
  345. MAP_BUTTON(18);
  346. MAP_BUTTON(19);
  347. MAP_BUTTON(20);
  348. MAP_BUTTON(21);
  349. MAP_BUTTON(22);
  350. MAP_BUTTON(23);
  351. MAP_BUTTON(24);
  352. MAP_BUTTON(25);
  353. MAP_BUTTON(26);
  354. MAP_BUTTON(27);
  355. MAP_BUTTON(28);
  356. MAP_BUTTON(29);
  357. default: break;
  358. }
  359. #undef MAP_BUTTON
  360. #undef MAP_KEY
  361. return false;
  362. }
  363. static bool code_to_str(int code, struct dstr *str)
  364. {
  365. #define MAP_GLYPH(c, g) \
  366. case c: dstr_from_wcs(str, (wchar_t[]){g, 0}); return true
  367. #define MAP_STR(c, s) case c: dstr_copy(str, s); return true
  368. switch (code) {
  369. MAP_GLYPH(kVK_Return, 0x21A9);
  370. MAP_GLYPH(kVK_Escape, 0x238B);
  371. MAP_GLYPH(kVK_Delete, 0x232B);
  372. MAP_GLYPH(kVK_Tab, 0x21e5);
  373. MAP_GLYPH(kVK_CapsLock, 0x21EA);
  374. MAP_GLYPH(kVK_ANSI_KeypadClear, 0x2327);
  375. MAP_GLYPH(kVK_ANSI_KeypadEnter, 0x2305);
  376. MAP_GLYPH(kVK_Help, 0x003F);
  377. MAP_GLYPH(kVK_Home, 0x2196);
  378. MAP_GLYPH(kVK_PageUp, 0x21de);
  379. MAP_GLYPH(kVK_ForwardDelete, 0x2326);
  380. MAP_GLYPH(kVK_End, 0x2198);
  381. MAP_GLYPH(kVK_PageDown, 0x21df);
  382. MAP_GLYPH(kVK_RightArrow, 0x2192);
  383. MAP_GLYPH(kVK_LeftArrow, 0x2190);
  384. MAP_GLYPH(kVK_DownArrow, 0x2193);
  385. MAP_GLYPH(kVK_UpArrow, 0x2191);
  386. MAP_STR (kVK_F1, "F1");
  387. MAP_STR (kVK_F2, "F2");
  388. MAP_STR (kVK_F3, "F3");
  389. MAP_STR (kVK_F4, "F4");
  390. MAP_STR (kVK_F5, "F5");
  391. MAP_STR (kVK_F6, "F6");
  392. MAP_STR (kVK_F7, "F7");
  393. MAP_STR (kVK_F8, "F8");
  394. MAP_STR (kVK_F9, "F9");
  395. MAP_STR (kVK_F10, "F10");
  396. MAP_STR (kVK_F11, "F11");
  397. MAP_STR (kVK_F12, "F12");
  398. MAP_STR (kVK_F13, "F13");
  399. MAP_STR (kVK_F14, "F14");
  400. MAP_STR (kVK_F15, "F15");
  401. MAP_STR (kVK_F16, "F16");
  402. MAP_STR (kVK_F17, "F17");
  403. MAP_STR (kVK_F18, "F18");
  404. MAP_STR (kVK_F19, "F19");
  405. MAP_STR (kVK_F20, "F20");
  406. MAP_GLYPH(kVK_Control, kControlUnicode);
  407. MAP_GLYPH(kVK_Shift, kShiftUnicode);
  408. MAP_GLYPH(kVK_Option, kOptionUnicode);
  409. MAP_GLYPH(kVK_Command, kCommandUnicode);
  410. MAP_GLYPH(kVK_RightControl, kControlUnicode);
  411. MAP_GLYPH(kVK_RightShift, kShiftUnicode);
  412. MAP_GLYPH(kVK_RightOption, kOptionUnicode);
  413. }
  414. #undef MAP_STR
  415. #undef MAP_GLYPH
  416. return false;
  417. }
  418. void obs_key_to_str(obs_key_t key, struct dstr *str)
  419. {
  420. if (localized_key_to_str(key, str))
  421. return;
  422. int code = obs_key_to_virtual_key(key);
  423. if (code_to_str(code, str))
  424. return;
  425. if (code == INVALID_KEY) {
  426. blog(LOG_ERROR, "hotkey-cocoa: Got invalid key while "
  427. "translating key '%d' (%s)",
  428. key, obs_key_to_name(key));
  429. goto err;
  430. }
  431. struct obs_hotkeys_platform *plat = NULL;
  432. if (obs) {
  433. pthread_mutex_lock(&obs->hotkeys.mutex);
  434. plat = obs->hotkeys.platform_context;
  435. hotkeys_retain(plat);
  436. pthread_mutex_unlock(&obs->hotkeys.mutex);
  437. }
  438. if (!plat) {
  439. blog(LOG_ERROR, "hotkey-cocoa: Could not get hotkey platform "
  440. "while translating key '%d' (%s)",
  441. key, obs_key_to_name(key));
  442. goto err;
  443. }
  444. const UniCharCount max_length = 16;
  445. UInt32 dead_key_state = 0;
  446. UniChar buffer[max_length];
  447. UniCharCount len = 0;
  448. OSStatus err = UCKeyTranslate(plat->layout,
  449. code,
  450. kUCKeyActionDown,
  451. 0x104, //caps lock for upper case letters
  452. LMGetKbdType(),
  453. kUCKeyTranslateNoDeadKeysBit,
  454. &dead_key_state,
  455. max_length,
  456. &len,
  457. buffer);
  458. if (err == noErr && len <= 0 && dead_key_state) {
  459. err = UCKeyTranslate(plat->layout,
  460. kVK_Space,
  461. kUCKeyActionDown,
  462. 0x104,
  463. LMGetKbdType(),
  464. kUCKeyTranslateNoDeadKeysBit,
  465. &dead_key_state,
  466. max_length,
  467. &len,
  468. buffer);
  469. }
  470. hotkeys_release(plat);
  471. if (err != noErr) {
  472. blog(LOG_ERROR, "hotkey-cocoa: Error while translating key '%d'"
  473. " (0x%x, %s) to string: %d", key, code,
  474. obs_key_to_name(key), err);
  475. goto err;
  476. }
  477. if (len == 0) {
  478. blog(LOG_ERROR, "hotkey-cocoa: Got 0 length string while "
  479. "translating '%d' (0x%x, %s) to string",
  480. key, code, obs_key_to_name(key));
  481. goto err;
  482. }
  483. CFStringRef string = CFStringCreateWithCharactersNoCopy(NULL,
  484. buffer, len, kCFAllocatorNull);
  485. if (!string) {
  486. blog(LOG_ERROR, "hotkey-cocoa: Could not create CFStringRef "
  487. "while translating '%d' (0x%x, %s) to string",
  488. key, code, obs_key_to_name(key));
  489. goto err;
  490. }
  491. if (!dstr_from_cfstring(str, string)) {
  492. blog(LOG_ERROR, "hotkey-cocoa: Could not translate CFStringRef "
  493. "to CString while translating '%d' (0x%x, %s)",
  494. key, code, obs_key_to_name(key));
  495. goto release;
  496. }
  497. CFRelease(string);
  498. return;
  499. release:
  500. CFRelease(string);
  501. err:
  502. dstr_copy(str, obs_key_to_name(key));
  503. }
  504. #define OBS_COCOA_MODIFIER_SIZE 7
  505. static void unichar_to_utf8(const UniChar *c, char *buff)
  506. {
  507. CFStringRef string = CFStringCreateWithCharactersNoCopy(NULL, c, 2,
  508. kCFAllocatorNull);
  509. if (!string) {
  510. blog(LOG_ERROR, "hotkey-cocoa: Could not create CFStringRef "
  511. "while populating modifier strings");
  512. return;
  513. }
  514. if (!CFStringGetCString(string, buff, OBS_COCOA_MODIFIER_SIZE,
  515. kCFStringEncodingUTF8))
  516. blog(LOG_ERROR, "hotkey-cocoa: Error while populating "
  517. " modifier string with glyph %d (0x%x)",
  518. c[0], c[0]);
  519. CFRelease(string);
  520. }
  521. static char ctrl_str[OBS_COCOA_MODIFIER_SIZE];
  522. static char opt_str[OBS_COCOA_MODIFIER_SIZE];
  523. static char shift_str[OBS_COCOA_MODIFIER_SIZE];
  524. static char cmd_str[OBS_COCOA_MODIFIER_SIZE];
  525. static void init_utf_8_strings(void)
  526. {
  527. const UniChar ctrl_uni[] = {kControlUnicode, 0};
  528. const UniChar opt_uni[] = {kOptionUnicode, 0};
  529. const UniChar shift_uni[] = {kShiftUnicode, 0};
  530. const UniChar cmd_uni[] = {kCommandUnicode, 0};
  531. unichar_to_utf8(ctrl_uni, ctrl_str);
  532. unichar_to_utf8(opt_uni, opt_str);
  533. unichar_to_utf8(shift_uni, shift_str);
  534. unichar_to_utf8(cmd_uni, cmd_str);
  535. }
  536. static pthread_once_t strings_token = PTHREAD_ONCE_INIT;
  537. void obs_key_combination_to_str(obs_key_combination_t key, struct dstr *str)
  538. {
  539. struct dstr key_str = {0};
  540. if (key.key != OBS_KEY_NONE)
  541. obs_key_to_str(key.key, &key_str);
  542. int res = pthread_once(&strings_token, init_utf_8_strings);
  543. if (res) {
  544. blog(LOG_ERROR, "hotkeys-cocoa: Error while translating "
  545. "modifiers %d (0x%x)", res, res);
  546. dstr_move(str, &key_str);
  547. return;
  548. }
  549. #define CHECK_MODIFIER(mod, str) ((key.modifiers & mod) ? str : "")
  550. dstr_printf(str, "%s%s%s%s%s",
  551. CHECK_MODIFIER(INTERACT_CONTROL_KEY, ctrl_str),
  552. CHECK_MODIFIER(INTERACT_ALT_KEY, opt_str),
  553. CHECK_MODIFIER(INTERACT_SHIFT_KEY, shift_str),
  554. CHECK_MODIFIER(INTERACT_COMMAND_KEY, cmd_str),
  555. key_str.len ? key_str.array : "");
  556. #undef CHECK_MODIFIER
  557. dstr_free(&key_str);
  558. }
  559. static inline CFDictionaryRef copy_device_mask(UInt32 page, UInt32 usage)
  560. {
  561. CFMutableDictionaryRef dict = CFDictionaryCreateMutable(
  562. kCFAllocatorDefault, 2,
  563. &kCFTypeDictionaryKeyCallBacks,
  564. &kCFTypeDictionaryValueCallBacks);
  565. CFNumberRef value;
  566. // Add the page value.
  567. value = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &page);
  568. CFDictionarySetValue(dict, CFSTR(kIOHIDDeviceUsagePageKey), value);
  569. CFRelease(value);
  570. // Add the usage value (which is only valid if page value exists).
  571. value = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &usage);
  572. CFDictionarySetValue(dict, CFSTR(kIOHIDDeviceUsageKey), value);
  573. CFRelease(value);
  574. return dict;
  575. }
  576. static CFSetRef copy_devices(obs_hotkeys_platform_t *plat,
  577. UInt32 page, UInt32 usage)
  578. {
  579. CFDictionaryRef mask = copy_device_mask(page, usage);
  580. IOHIDManagerSetDeviceMatching(plat->manager, mask);
  581. CFRelease(mask);
  582. CFSetRef devices = IOHIDManagerCopyDevices(plat->manager);
  583. if (!devices)
  584. return NULL;
  585. if (CFSetGetCount(devices) < 1) {
  586. CFRelease(devices);
  587. return NULL;
  588. }
  589. return devices;
  590. }
  591. static UInt16 usage_to_carbon(UInt32 usage)
  592. {
  593. switch (usage)
  594. {
  595. case kHIDUsage_KeyboardErrorRollOver: return INVALID_KEY;
  596. case kHIDUsage_KeyboardPOSTFail: return INVALID_KEY;
  597. case kHIDUsage_KeyboardErrorUndefined: return INVALID_KEY;
  598. case kHIDUsage_KeyboardA: return kVK_ANSI_A;
  599. case kHIDUsage_KeyboardB: return kVK_ANSI_B;
  600. case kHIDUsage_KeyboardC: return kVK_ANSI_C;
  601. case kHIDUsage_KeyboardD: return kVK_ANSI_D;
  602. case kHIDUsage_KeyboardE: return kVK_ANSI_E;
  603. case kHIDUsage_KeyboardF: return kVK_ANSI_F;
  604. case kHIDUsage_KeyboardG: return kVK_ANSI_G;
  605. case kHIDUsage_KeyboardH: return kVK_ANSI_H;
  606. case kHIDUsage_KeyboardI: return kVK_ANSI_I;
  607. case kHIDUsage_KeyboardJ: return kVK_ANSI_J;
  608. case kHIDUsage_KeyboardK: return kVK_ANSI_K;
  609. case kHIDUsage_KeyboardL: return kVK_ANSI_L;
  610. case kHIDUsage_KeyboardM: return kVK_ANSI_M;
  611. case kHIDUsage_KeyboardN: return kVK_ANSI_N;
  612. case kHIDUsage_KeyboardO: return kVK_ANSI_O;
  613. case kHIDUsage_KeyboardP: return kVK_ANSI_P;
  614. case kHIDUsage_KeyboardQ: return kVK_ANSI_Q;
  615. case kHIDUsage_KeyboardR: return kVK_ANSI_R;
  616. case kHIDUsage_KeyboardS: return kVK_ANSI_S;
  617. case kHIDUsage_KeyboardT: return kVK_ANSI_T;
  618. case kHIDUsage_KeyboardU: return kVK_ANSI_U;
  619. case kHIDUsage_KeyboardV: return kVK_ANSI_V;
  620. case kHIDUsage_KeyboardW: return kVK_ANSI_W;
  621. case kHIDUsage_KeyboardX: return kVK_ANSI_X;
  622. case kHIDUsage_KeyboardY: return kVK_ANSI_Y;
  623. case kHIDUsage_KeyboardZ: return kVK_ANSI_Z;
  624. case kHIDUsage_Keyboard1: return kVK_ANSI_1;
  625. case kHIDUsage_Keyboard2: return kVK_ANSI_2;
  626. case kHIDUsage_Keyboard3: return kVK_ANSI_3;
  627. case kHIDUsage_Keyboard4: return kVK_ANSI_4;
  628. case kHIDUsage_Keyboard5: return kVK_ANSI_5;
  629. case kHIDUsage_Keyboard6: return kVK_ANSI_6;
  630. case kHIDUsage_Keyboard7: return kVK_ANSI_7;
  631. case kHIDUsage_Keyboard8: return kVK_ANSI_8;
  632. case kHIDUsage_Keyboard9: return kVK_ANSI_9;
  633. case kHIDUsage_Keyboard0: return kVK_ANSI_0;
  634. case kHIDUsage_KeyboardReturnOrEnter: return kVK_Return;
  635. case kHIDUsage_KeyboardEscape: return kVK_Escape;
  636. case kHIDUsage_KeyboardDeleteOrBackspace: return kVK_Delete;
  637. case kHIDUsage_KeyboardTab: return kVK_Tab;
  638. case kHIDUsage_KeyboardSpacebar: return kVK_Space;
  639. case kHIDUsage_KeyboardHyphen: return kVK_ANSI_Minus;
  640. case kHIDUsage_KeyboardEqualSign: return kVK_ANSI_Equal;
  641. case kHIDUsage_KeyboardOpenBracket: return kVK_ANSI_LeftBracket;
  642. case kHIDUsage_KeyboardCloseBracket: return kVK_ANSI_RightBracket;
  643. case kHIDUsage_KeyboardBackslash: return kVK_ANSI_Backslash;
  644. case kHIDUsage_KeyboardNonUSPound: return INVALID_KEY;
  645. case kHIDUsage_KeyboardSemicolon: return kVK_ANSI_Semicolon;
  646. case kHIDUsage_KeyboardQuote: return kVK_ANSI_Quote;
  647. case kHIDUsage_KeyboardGraveAccentAndTilde: return kVK_ANSI_Grave;
  648. case kHIDUsage_KeyboardComma: return kVK_ANSI_Comma;
  649. case kHIDUsage_KeyboardPeriod: return kVK_ANSI_Period;
  650. case kHIDUsage_KeyboardSlash: return kVK_ANSI_Slash;
  651. case kHIDUsage_KeyboardCapsLock: return kVK_CapsLock;
  652. case kHIDUsage_KeyboardF1: return kVK_F1;
  653. case kHIDUsage_KeyboardF2: return kVK_F2;
  654. case kHIDUsage_KeyboardF3: return kVK_F3;
  655. case kHIDUsage_KeyboardF4: return kVK_F4;
  656. case kHIDUsage_KeyboardF5: return kVK_F5;
  657. case kHIDUsage_KeyboardF6: return kVK_F6;
  658. case kHIDUsage_KeyboardF7: return kVK_F7;
  659. case kHIDUsage_KeyboardF8: return kVK_F8;
  660. case kHIDUsage_KeyboardF9: return kVK_F9;
  661. case kHIDUsage_KeyboardF10: return kVK_F10;
  662. case kHIDUsage_KeyboardF11: return kVK_F11;
  663. case kHIDUsage_KeyboardF12: return kVK_F12;
  664. case kHIDUsage_KeyboardPrintScreen: return INVALID_KEY;
  665. case kHIDUsage_KeyboardScrollLock: return INVALID_KEY;
  666. case kHIDUsage_KeyboardPause: return INVALID_KEY;
  667. case kHIDUsage_KeyboardInsert: return kVK_Help;
  668. case kHIDUsage_KeyboardHome: return kVK_Home;
  669. case kHIDUsage_KeyboardPageUp: return kVK_PageUp;
  670. case kHIDUsage_KeyboardDeleteForward: return kVK_ForwardDelete;
  671. case kHIDUsage_KeyboardEnd: return kVK_End;
  672. case kHIDUsage_KeyboardPageDown: return kVK_PageDown;
  673. case kHIDUsage_KeyboardRightArrow: return kVK_RightArrow;
  674. case kHIDUsage_KeyboardLeftArrow: return kVK_LeftArrow;
  675. case kHIDUsage_KeyboardDownArrow: return kVK_DownArrow;
  676. case kHIDUsage_KeyboardUpArrow: return kVK_UpArrow;
  677. case kHIDUsage_KeypadNumLock: return kVK_ANSI_KeypadClear;
  678. case kHIDUsage_KeypadSlash: return kVK_ANSI_KeypadDivide;
  679. case kHIDUsage_KeypadAsterisk: return kVK_ANSI_KeypadMultiply;
  680. case kHIDUsage_KeypadHyphen: return kVK_ANSI_KeypadMinus;
  681. case kHIDUsage_KeypadPlus: return kVK_ANSI_KeypadPlus;
  682. case kHIDUsage_KeypadEnter: return kVK_ANSI_KeypadEnter;
  683. case kHIDUsage_Keypad1: return kVK_ANSI_Keypad1;
  684. case kHIDUsage_Keypad2: return kVK_ANSI_Keypad2;
  685. case kHIDUsage_Keypad3: return kVK_ANSI_Keypad3;
  686. case kHIDUsage_Keypad4: return kVK_ANSI_Keypad4;
  687. case kHIDUsage_Keypad5: return kVK_ANSI_Keypad5;
  688. case kHIDUsage_Keypad6: return kVK_ANSI_Keypad6;
  689. case kHIDUsage_Keypad7: return kVK_ANSI_Keypad7;
  690. case kHIDUsage_Keypad8: return kVK_ANSI_Keypad8;
  691. case kHIDUsage_Keypad9: return kVK_ANSI_Keypad9;
  692. case kHIDUsage_Keypad0: return kVK_ANSI_Keypad0;
  693. case kHIDUsage_KeypadPeriod: return kVK_ANSI_KeypadDecimal;
  694. case kHIDUsage_KeyboardNonUSBackslash: return INVALID_KEY;
  695. case kHIDUsage_KeyboardApplication: return kVK_F13;
  696. case kHIDUsage_KeyboardPower: return INVALID_KEY;
  697. case kHIDUsage_KeypadEqualSign: return kVK_ANSI_KeypadEquals;
  698. case kHIDUsage_KeyboardF13: return kVK_F13;
  699. case kHIDUsage_KeyboardF14: return kVK_F14;
  700. case kHIDUsage_KeyboardF15: return kVK_F15;
  701. case kHIDUsage_KeyboardF16: return kVK_F16;
  702. case kHIDUsage_KeyboardF17: return kVK_F17;
  703. case kHIDUsage_KeyboardF18: return kVK_F18;
  704. case kHIDUsage_KeyboardF19: return kVK_F19;
  705. case kHIDUsage_KeyboardF20: return kVK_F20;
  706. case kHIDUsage_KeyboardF21: return INVALID_KEY;
  707. case kHIDUsage_KeyboardF22: return INVALID_KEY;
  708. case kHIDUsage_KeyboardF23: return INVALID_KEY;
  709. case kHIDUsage_KeyboardF24: return INVALID_KEY;
  710. case kHIDUsage_KeyboardExecute: return INVALID_KEY;
  711. case kHIDUsage_KeyboardHelp: return INVALID_KEY;
  712. case kHIDUsage_KeyboardMenu: return 0x7F;
  713. case kHIDUsage_KeyboardSelect: return kVK_ANSI_KeypadEnter;
  714. case kHIDUsage_KeyboardStop: return INVALID_KEY;
  715. case kHIDUsage_KeyboardAgain: return INVALID_KEY;
  716. case kHIDUsage_KeyboardUndo: return INVALID_KEY;
  717. case kHIDUsage_KeyboardCut: return INVALID_KEY;
  718. case kHIDUsage_KeyboardCopy: return INVALID_KEY;
  719. case kHIDUsage_KeyboardPaste: return INVALID_KEY;
  720. case kHIDUsage_KeyboardFind: return INVALID_KEY;
  721. case kHIDUsage_KeyboardMute: return kVK_Mute;
  722. case kHIDUsage_KeyboardVolumeUp: return kVK_VolumeUp;
  723. case kHIDUsage_KeyboardVolumeDown: return kVK_VolumeDown;
  724. case kHIDUsage_KeyboardLockingCapsLock: return INVALID_KEY;
  725. case kHIDUsage_KeyboardLockingNumLock: return INVALID_KEY;
  726. case kHIDUsage_KeyboardLockingScrollLock: return INVALID_KEY;
  727. case kHIDUsage_KeypadComma: return INVALID_KEY;
  728. case kHIDUsage_KeypadEqualSignAS400: return INVALID_KEY;
  729. case kHIDUsage_KeyboardInternational1: return INVALID_KEY;
  730. case kHIDUsage_KeyboardInternational2: return INVALID_KEY;
  731. case kHIDUsage_KeyboardInternational3: return INVALID_KEY;
  732. case kHIDUsage_KeyboardInternational4: return INVALID_KEY;
  733. case kHIDUsage_KeyboardInternational5: return INVALID_KEY;
  734. case kHIDUsage_KeyboardInternational6: return INVALID_KEY;
  735. case kHIDUsage_KeyboardInternational7: return INVALID_KEY;
  736. case kHIDUsage_KeyboardInternational8: return INVALID_KEY;
  737. case kHIDUsage_KeyboardInternational9: return INVALID_KEY;
  738. case kHIDUsage_KeyboardLANG1: return INVALID_KEY;
  739. case kHIDUsage_KeyboardLANG2: return INVALID_KEY;
  740. case kHIDUsage_KeyboardLANG3: return INVALID_KEY;
  741. case kHIDUsage_KeyboardLANG4: return INVALID_KEY;
  742. case kHIDUsage_KeyboardLANG5: return INVALID_KEY;
  743. case kHIDUsage_KeyboardLANG6: return INVALID_KEY;
  744. case kHIDUsage_KeyboardLANG7: return INVALID_KEY;
  745. case kHIDUsage_KeyboardLANG8: return INVALID_KEY;
  746. case kHIDUsage_KeyboardLANG9: return INVALID_KEY;
  747. case kHIDUsage_KeyboardAlternateErase: return INVALID_KEY;
  748. case kHIDUsage_KeyboardSysReqOrAttention: return INVALID_KEY;
  749. case kHIDUsage_KeyboardCancel: return INVALID_KEY;
  750. case kHIDUsage_KeyboardClear: return INVALID_KEY;
  751. case kHIDUsage_KeyboardPrior: return INVALID_KEY;
  752. case kHIDUsage_KeyboardReturn: return INVALID_KEY;
  753. case kHIDUsage_KeyboardSeparator: return INVALID_KEY;
  754. case kHIDUsage_KeyboardOut: return INVALID_KEY;
  755. case kHIDUsage_KeyboardOper: return INVALID_KEY;
  756. case kHIDUsage_KeyboardClearOrAgain: return INVALID_KEY;
  757. case kHIDUsage_KeyboardCrSelOrProps: return INVALID_KEY;
  758. case kHIDUsage_KeyboardExSel: return INVALID_KEY;
  759. /* 0xa5-0xdf Reserved */
  760. case kHIDUsage_KeyboardLeftControl: return kVK_Control;
  761. case kHIDUsage_KeyboardLeftShift: return kVK_Shift;
  762. case kHIDUsage_KeyboardLeftAlt: return kVK_Option;
  763. case kHIDUsage_KeyboardLeftGUI: return kVK_Command;
  764. case kHIDUsage_KeyboardRightControl: return kVK_RightControl;
  765. case kHIDUsage_KeyboardRightShift: return kVK_RightShift;
  766. case kHIDUsage_KeyboardRightAlt: return kVK_RightOption;
  767. case kHIDUsage_KeyboardRightGUI: return 0x36; //??
  768. /* 0xe8-0xffff Reserved */
  769. case kHIDUsage_Keyboard_Reserved: return INVALID_KEY;
  770. default: return INVALID_KEY;
  771. }
  772. return INVALID_KEY;
  773. }
  774. obs_key_t obs_key_from_virtual_key(int code)
  775. {
  776. switch (code) {
  777. case kVK_ANSI_A: return OBS_KEY_A;
  778. case kVK_ANSI_B: return OBS_KEY_B;
  779. case kVK_ANSI_C: return OBS_KEY_C;
  780. case kVK_ANSI_D: return OBS_KEY_D;
  781. case kVK_ANSI_E: return OBS_KEY_E;
  782. case kVK_ANSI_F: return OBS_KEY_F;
  783. case kVK_ANSI_G: return OBS_KEY_G;
  784. case kVK_ANSI_H: return OBS_KEY_H;
  785. case kVK_ANSI_I: return OBS_KEY_I;
  786. case kVK_ANSI_J: return OBS_KEY_J;
  787. case kVK_ANSI_K: return OBS_KEY_K;
  788. case kVK_ANSI_L: return OBS_KEY_L;
  789. case kVK_ANSI_M: return OBS_KEY_M;
  790. case kVK_ANSI_N: return OBS_KEY_N;
  791. case kVK_ANSI_O: return OBS_KEY_O;
  792. case kVK_ANSI_P: return OBS_KEY_P;
  793. case kVK_ANSI_Q: return OBS_KEY_Q;
  794. case kVK_ANSI_R: return OBS_KEY_R;
  795. case kVK_ANSI_S: return OBS_KEY_S;
  796. case kVK_ANSI_T: return OBS_KEY_T;
  797. case kVK_ANSI_U: return OBS_KEY_U;
  798. case kVK_ANSI_V: return OBS_KEY_V;
  799. case kVK_ANSI_W: return OBS_KEY_W;
  800. case kVK_ANSI_X: return OBS_KEY_X;
  801. case kVK_ANSI_Y: return OBS_KEY_Y;
  802. case kVK_ANSI_Z: return OBS_KEY_Z;
  803. case kVK_ANSI_1: return OBS_KEY_1;
  804. case kVK_ANSI_2: return OBS_KEY_2;
  805. case kVK_ANSI_3: return OBS_KEY_3;
  806. case kVK_ANSI_4: return OBS_KEY_4;
  807. case kVK_ANSI_5: return OBS_KEY_5;
  808. case kVK_ANSI_6: return OBS_KEY_6;
  809. case kVK_ANSI_7: return OBS_KEY_7;
  810. case kVK_ANSI_8: return OBS_KEY_8;
  811. case kVK_ANSI_9: return OBS_KEY_9;
  812. case kVK_ANSI_0: return OBS_KEY_0;
  813. case kVK_Return: return OBS_KEY_RETURN;
  814. case kVK_Escape: return OBS_KEY_ESCAPE;
  815. case kVK_Delete: return OBS_KEY_BACKSPACE;
  816. case kVK_Tab: return OBS_KEY_TAB;
  817. case kVK_Space: return OBS_KEY_SPACE;
  818. case kVK_ANSI_Minus: return OBS_KEY_MINUS;
  819. case kVK_ANSI_Equal: return OBS_KEY_EQUAL;
  820. case kVK_ANSI_LeftBracket: return OBS_KEY_BRACKETLEFT;
  821. case kVK_ANSI_RightBracket: return OBS_KEY_BRACKETRIGHT;
  822. case kVK_ANSI_Backslash: return OBS_KEY_BACKSLASH;
  823. case kVK_ANSI_Semicolon: return OBS_KEY_SEMICOLON;
  824. case kVK_ANSI_Quote: return OBS_KEY_QUOTE;
  825. case kVK_ANSI_Grave: return OBS_KEY_DEAD_GRAVE;
  826. case kVK_ANSI_Comma: return OBS_KEY_COMMA;
  827. case kVK_ANSI_Period: return OBS_KEY_PERIOD;
  828. case kVK_ANSI_Slash: return OBS_KEY_SLASH;
  829. case kVK_CapsLock: return OBS_KEY_CAPSLOCK;
  830. case kVK_ISO_Section: return OBS_KEY_SECTION;
  831. case kVK_F1: return OBS_KEY_F1;
  832. case kVK_F2: return OBS_KEY_F2;
  833. case kVK_F3: return OBS_KEY_F3;
  834. case kVK_F4: return OBS_KEY_F4;
  835. case kVK_F5: return OBS_KEY_F5;
  836. case kVK_F6: return OBS_KEY_F6;
  837. case kVK_F7: return OBS_KEY_F7;
  838. case kVK_F8: return OBS_KEY_F8;
  839. case kVK_F9: return OBS_KEY_F9;
  840. case kVK_F10: return OBS_KEY_F10;
  841. case kVK_F11: return OBS_KEY_F11;
  842. case kVK_F12: return OBS_KEY_F12;
  843. case kVK_Help: return OBS_KEY_HELP;
  844. case kVK_Home: return OBS_KEY_HOME;
  845. case kVK_PageUp: return OBS_KEY_PAGEUP;
  846. case kVK_ForwardDelete: return OBS_KEY_DELETE;
  847. case kVK_End: return OBS_KEY_END;
  848. case kVK_PageDown: return OBS_KEY_PAGEDOWN;
  849. case kVK_RightArrow: return OBS_KEY_RIGHT;
  850. case kVK_LeftArrow: return OBS_KEY_LEFT;
  851. case kVK_DownArrow: return OBS_KEY_DOWN;
  852. case kVK_UpArrow: return OBS_KEY_UP;
  853. case kVK_ANSI_KeypadClear: return OBS_KEY_CLEAR;
  854. case kVK_ANSI_KeypadDivide: return OBS_KEY_NUMSLASH;
  855. case kVK_ANSI_KeypadMultiply: return OBS_KEY_NUMASTERISK;
  856. case kVK_ANSI_KeypadMinus: return OBS_KEY_NUMMINUS;
  857. case kVK_ANSI_KeypadPlus: return OBS_KEY_NUMPLUS;
  858. case kVK_ANSI_KeypadEnter: return OBS_KEY_ENTER;
  859. case kVK_ANSI_Keypad1: return OBS_KEY_NUM1;
  860. case kVK_ANSI_Keypad2: return OBS_KEY_NUM2;
  861. case kVK_ANSI_Keypad3: return OBS_KEY_NUM3;
  862. case kVK_ANSI_Keypad4: return OBS_KEY_NUM4;
  863. case kVK_ANSI_Keypad5: return OBS_KEY_NUM5;
  864. case kVK_ANSI_Keypad6: return OBS_KEY_NUM6;
  865. case kVK_ANSI_Keypad7: return OBS_KEY_NUM7;
  866. case kVK_ANSI_Keypad8: return OBS_KEY_NUM8;
  867. case kVK_ANSI_Keypad9: return OBS_KEY_NUM9;
  868. case kVK_ANSI_Keypad0: return OBS_KEY_NUM0;
  869. case kVK_ANSI_KeypadDecimal: return OBS_KEY_NUMPERIOD;
  870. case kVK_ANSI_KeypadEquals: return OBS_KEY_NUMEQUAL;
  871. case kVK_F13: return OBS_KEY_F13;
  872. case kVK_F14: return OBS_KEY_F14;
  873. case kVK_F15: return OBS_KEY_F15;
  874. case kVK_F16: return OBS_KEY_F16;
  875. case kVK_F17: return OBS_KEY_F17;
  876. case kVK_F18: return OBS_KEY_F18;
  877. case kVK_F19: return OBS_KEY_F19;
  878. case kVK_F20: return OBS_KEY_F20;
  879. case kVK_Control: return OBS_KEY_CONTROL;
  880. case kVK_Shift: return OBS_KEY_SHIFT;
  881. case kVK_Option: return OBS_KEY_ALT;
  882. case kVK_Command: return OBS_KEY_META;
  883. case kVK_RightControl: return OBS_KEY_CONTROL;
  884. case kVK_RightShift: return OBS_KEY_SHIFT;
  885. case kVK_RightOption: return OBS_KEY_ALT;
  886. case 0x36: return OBS_KEY_META;
  887. case kVK_Function:
  888. case kVK_Mute:
  889. case kVK_VolumeDown:
  890. case kVK_VolumeUp:
  891. break;
  892. }
  893. return OBS_KEY_NONE;
  894. }
  895. static inline void load_key(obs_hotkeys_platform_t *plat, IOHIDElementRef key)
  896. {
  897. UInt32 usage_code = IOHIDElementGetUsage(key);
  898. UInt16 carbon_code = usage_to_carbon(usage_code);
  899. if (carbon_code == INVALID_KEY) return;
  900. obs_key_t obs_key = obs_key_from_virtual_key(carbon_code);
  901. if (obs_key == OBS_KEY_NONE)
  902. return;
  903. da_push_back(plat->keys[obs_key], &key);
  904. CFRetain(*(IOHIDElementRef*)da_end(plat->keys[obs_key]));
  905. }
  906. static inline void load_keyboard(obs_hotkeys_platform_t *plat,
  907. IOHIDDeviceRef keyboard)
  908. {
  909. CFArrayRef keys = IOHIDDeviceCopyMatchingElements(keyboard, NULL,
  910. kIOHIDOptionsTypeNone);
  911. if (!keys) {
  912. blog(LOG_ERROR, "hotkeys-cocoa: Getting keyboard keys failed");
  913. return;
  914. }
  915. CFIndex count = CFArrayGetCount(keys);
  916. if (!count) {
  917. blog(LOG_ERROR, "hotkeys-cocoa: Keyboard has no keys");
  918. CFRelease(keys);
  919. return;
  920. }
  921. for (CFIndex i = 0; i < count; i++) {
  922. IOHIDElementRef key =
  923. (IOHIDElementRef)CFArrayGetValueAtIndex(keys, i);
  924. // Skip non-matching keys elements
  925. if (IOHIDElementGetUsagePage(key) != kHIDPage_KeyboardOrKeypad)
  926. continue;
  927. load_key(plat, key);
  928. }
  929. CFRelease(keys);
  930. }
  931. static bool init_keyboard(obs_hotkeys_platform_t *plat)
  932. {
  933. CFSetRef keyboards = copy_devices(plat, kHIDPage_GenericDesktop,
  934. kHIDUsage_GD_Keyboard);
  935. if (!keyboards)
  936. return false;
  937. CFIndex count = CFSetGetCount(keyboards);
  938. CFTypeRef devices[count];
  939. CFSetGetValues(keyboards, devices);
  940. for (CFIndex i = 0; i < count; i++)
  941. load_keyboard(plat, (IOHIDDeviceRef)devices[i]);
  942. CFRelease(keyboards);
  943. return true;
  944. }
  945. static inline void free_hotkeys_platform(obs_hotkeys_platform_t *plat)
  946. {
  947. if (!plat)
  948. return;
  949. if (plat->tis) {
  950. CFRelease(plat->tis);
  951. plat->tis = NULL;
  952. }
  953. if (plat->layout_data) {
  954. CFRelease(plat->layout_data);
  955. plat->layout_data = NULL;
  956. }
  957. if (plat->manager) {
  958. CFRelease(plat->manager);
  959. plat->manager = NULL;
  960. }
  961. for (size_t i = 0; i < OBS_KEY_LAST_VALUE; i++) {
  962. for (size_t j = 0; j < plat->keys[i].num; j++)
  963. CFRelease(plat->keys[i].array[j]);
  964. da_free(plat->keys[i]);
  965. }
  966. bfree(plat);
  967. }
  968. static bool log_layout_name(TISInputSourceRef tis)
  969. {
  970. struct dstr layout_name = {0};
  971. CFStringRef sid = (CFStringRef)TISGetInputSourceProperty(tis,
  972. kTISPropertyInputSourceID);
  973. if (!sid) {
  974. blog(LOG_ERROR, "hotkeys-cocoa: Failed getting InputSourceID");
  975. return false;
  976. }
  977. if (!dstr_from_cfstring(&layout_name, sid)) {
  978. blog(LOG_ERROR, "hotkeys-cocoa: Could not convert InputSourceID"
  979. " to CString");
  980. goto fail;
  981. }
  982. blog(LOG_INFO, "hotkeys-cocoa: Using layout '%s'", layout_name.array);
  983. dstr_free(&layout_name);
  984. return true;
  985. fail:
  986. dstr_free(&layout_name);
  987. return false;
  988. }
  989. static bool init_hotkeys_platform(obs_hotkeys_platform_t **plat_)
  990. {
  991. if (!plat_)
  992. return false;
  993. *plat_ = bzalloc(sizeof(obs_hotkeys_platform_t));
  994. obs_hotkeys_platform_t *plat = *plat_;
  995. if (!plat) {
  996. *plat_ = NULL;
  997. return false;
  998. }
  999. plat->tis = TISCopyCurrentKeyboardLayoutInputSource();
  1000. plat->layout_data = (CFDataRef)TISGetInputSourceProperty(plat->tis,
  1001. kTISPropertyUnicodeKeyLayoutData);
  1002. if (!plat->layout_data) {
  1003. blog(LOG_ERROR, "hotkeys-cocoa: Failed getting LayoutData");
  1004. goto fail;
  1005. }
  1006. CFRetain(plat->layout_data);
  1007. plat->layout = (UCKeyboardLayout*)CFDataGetBytePtr(plat->layout_data);
  1008. plat->manager = IOHIDManagerCreate(kCFAllocatorDefault,
  1009. kIOHIDOptionsTypeNone);
  1010. IOReturn openStatus = IOHIDManagerOpen(plat->manager,
  1011. kIOHIDOptionsTypeNone);
  1012. if (openStatus != kIOReturnSuccess) {
  1013. blog(LOG_ERROR, "hotkeys-cocoa: Failed opening HIDManager");
  1014. goto fail;
  1015. }
  1016. init_keyboard(plat);
  1017. return true;
  1018. fail:
  1019. hotkeys_release(plat);
  1020. *plat_ = NULL;
  1021. return false;
  1022. }
  1023. static void input_method_changed(CFNotificationCenterRef nc, void *observer,
  1024. CFStringRef name, const void *object, CFDictionaryRef user_info)
  1025. {
  1026. UNUSED_PARAMETER(nc);
  1027. UNUSED_PARAMETER(name);
  1028. UNUSED_PARAMETER(object);
  1029. UNUSED_PARAMETER(user_info);
  1030. struct obs_core_hotkeys *hotkeys = observer;
  1031. obs_hotkeys_platform_t *new_plat;
  1032. if (init_hotkeys_platform(&new_plat)) {
  1033. obs_hotkeys_platform_t *plat;
  1034. pthread_mutex_lock(&hotkeys->mutex);
  1035. plat = hotkeys->platform_context;
  1036. if (new_plat && plat &&
  1037. new_plat->layout_data == plat->layout_data) {
  1038. pthread_mutex_unlock(&hotkeys->mutex);
  1039. hotkeys_release(new_plat);
  1040. return;
  1041. }
  1042. hotkeys->platform_context = new_plat;
  1043. if (new_plat)
  1044. log_layout_name(new_plat->tis);
  1045. pthread_mutex_unlock(&hotkeys->mutex);
  1046. calldata_t params = {0};
  1047. signal_handler_signal(hotkeys->signals,
  1048. "hotkey_layout_change", &params);
  1049. if (plat)
  1050. hotkeys_release(plat);
  1051. }
  1052. }
  1053. bool obs_hotkeys_platform_init(struct obs_core_hotkeys *hotkeys)
  1054. {
  1055. CFNotificationCenterAddObserver(
  1056. CFNotificationCenterGetDistributedCenter(),
  1057. hotkeys, input_method_changed,
  1058. kTISNotifySelectedKeyboardInputSourceChanged, NULL,
  1059. CFNotificationSuspensionBehaviorDeliverImmediately);
  1060. input_method_changed(NULL, hotkeys, NULL, NULL, NULL);
  1061. return hotkeys->platform_context != NULL;
  1062. }
  1063. void obs_hotkeys_platform_free(struct obs_core_hotkeys *hotkeys)
  1064. {
  1065. CFNotificationCenterRemoveEveryObserver(
  1066. CFNotificationCenterGetDistributedCenter(),
  1067. hotkeys);
  1068. hotkeys_release(hotkeys->platform_context);
  1069. }
  1070. typedef unsigned long NSUInteger;
  1071. static bool mouse_button_pressed(obs_key_t key, bool *pressed)
  1072. {
  1073. int button = 0;
  1074. switch (key) {
  1075. #define MAP_BUTTON(n) case OBS_KEY_MOUSE ## n: button = n - 1; break
  1076. MAP_BUTTON(1);
  1077. MAP_BUTTON(2);
  1078. MAP_BUTTON(3);
  1079. MAP_BUTTON(4);
  1080. MAP_BUTTON(5);
  1081. MAP_BUTTON(6);
  1082. MAP_BUTTON(7);
  1083. MAP_BUTTON(8);
  1084. MAP_BUTTON(9);
  1085. MAP_BUTTON(10);
  1086. MAP_BUTTON(11);
  1087. MAP_BUTTON(12);
  1088. MAP_BUTTON(13);
  1089. MAP_BUTTON(14);
  1090. MAP_BUTTON(15);
  1091. MAP_BUTTON(16);
  1092. MAP_BUTTON(17);
  1093. MAP_BUTTON(18);
  1094. MAP_BUTTON(19);
  1095. MAP_BUTTON(20);
  1096. MAP_BUTTON(21);
  1097. MAP_BUTTON(22);
  1098. MAP_BUTTON(23);
  1099. MAP_BUTTON(24);
  1100. MAP_BUTTON(25);
  1101. MAP_BUTTON(26);
  1102. MAP_BUTTON(27);
  1103. MAP_BUTTON(28);
  1104. MAP_BUTTON(29);
  1105. break;
  1106. #undef MAP_BUTTON
  1107. default:
  1108. return false;
  1109. }
  1110. Class NSEvent = objc_getClass("NSEvent");
  1111. SEL pressedMouseButtons = sel_registerName("pressedMouseButtons");
  1112. NSUInteger buttons = (NSUInteger)objc_msgSend((id)NSEvent,
  1113. pressedMouseButtons);
  1114. *pressed = (buttons & (1 << button)) != 0;
  1115. return true;
  1116. }
  1117. bool obs_hotkeys_platform_is_pressed(obs_hotkeys_platform_t *plat,
  1118. obs_key_t key)
  1119. {
  1120. bool mouse_pressed = false;
  1121. if (mouse_button_pressed(key, &mouse_pressed))
  1122. return mouse_pressed;
  1123. if (!plat)
  1124. return false;
  1125. if (key >= OBS_KEY_LAST_VALUE)
  1126. return false;
  1127. for (size_t i = 0; i < plat->keys[key].num;) {
  1128. IOHIDElementRef element = plat->keys[key].array[i];
  1129. IOHIDValueRef value = 0;
  1130. IOHIDDeviceRef device = IOHIDElementGetDevice(element);
  1131. IOHIDDeviceGetValue(device, element, &value);
  1132. if (!value) {
  1133. CFRelease(element);
  1134. da_erase(plat->keys[key], i);
  1135. continue;
  1136. }
  1137. if (IOHIDValueGetIntegerValue(value) == 1)
  1138. return true;
  1139. i += 1;
  1140. }
  1141. return false;
  1142. }