diff --git a/src/platform/macos/input.cpp b/src/platform/macos/input.cpp index 11c6d228421..554c51cb16e 100644 --- a/src/platform/macos/input.cpp +++ b/src/platform/macos/input.cpp @@ -6,6 +6,8 @@ #include #include +#include "misc.h" + #include "src/logging.h" #include "src/platform/common.h" #include "src/utility.h" @@ -219,6 +221,11 @@ const KeyCodeMap kKeyCodesMap[] = { }; // clang-format on + /** + * Used to avoid spamming permission requests when the user receives an input event + */ + bool accessibility_permission_requested; + int keysym(int keycode) { KeyCodeMap key_map {}; @@ -235,12 +242,36 @@ const KeyCodeMap kKeyCodesMap[] = { return temp_map->mac_keycode; } + std::string + default_accessibility_log_msg() { + return "Accessibility permission is not enabled," + " please enable sunshine in " + "[System Settings > Privacy & Security > Privacy > Accessibility]" + ", then please restart Sunshine for it to take effect"; + } + + void + print_accessibility_status(const bool is_keyboard_event, const bool release) { + if (!release) return; + + if (!has_accessibility_permission()) { + if (!accessibility_permission_requested) { + accessibility_permission_requested = true; + request_accessibility_permission(); + } + BOOST_LOG(info) << "Received " << (is_keyboard_event ? "keyboard" : "mouse") << " event but " + << default_accessibility_log_msg(); + } + } + void keyboard(input_t &input, uint16_t modcode, bool release, uint8_t flags) { auto key = keysym(modcode); BOOST_LOG(debug) << "got keycode: 0x"sv << std::hex << modcode << ", translated to: 0x" << std::hex << key << ", release:" << release; + print_accessibility_status(true, release); + if (key < 0) { return; } @@ -412,6 +443,8 @@ const KeyCodeMap kKeyCodesMap[] = { BOOST_LOG(warning) << "Unsupported mouse button for MacOS: "sv << button; return; } + + print_accessibility_status(false, release); mouse->mouse_down[mac_button] = !release; @@ -511,6 +544,11 @@ const KeyCodeMap kKeyCodesMap[] = { auto macos_input = (macos_input_t *) result.get(); + accessibility_permission_requested = false; + if (request_accessibility_permission()) { + BOOST_LOG(info) << default_accessibility_log_msg() << ", to allow mouse clicks and keyboard inputs."; + } + // Default to main display macos_input->display = CGMainDisplayID(); diff --git a/src/platform/macos/misc.h b/src/platform/macos/misc.h index ca74f0ea478..d03674d9f3d 100644 --- a/src/platform/macos/misc.h +++ b/src/platform/macos/misc.h @@ -8,6 +8,22 @@ #include +namespace platf { + /** + * Prompts the user for Accessibility permission + * @return returns true if requested permission, false if already has permission + */ + bool + request_accessibility_permission(); + + /** + * Checks for Accessibility permission + * @return returns true if sunshine has Accessibility permission enabled + */ + bool + has_accessibility_permission(); +} + namespace dyn { typedef void (*apiproc)(); diff --git a/src/platform/macos/misc.mm b/src/platform/macos/misc.mm index 20c2247e049..07a839cfc7f 100644 --- a/src/platform/macos/misc.mm +++ b/src/platform/macos/misc.mm @@ -8,6 +8,8 @@ #define __APPLE_USE_RFC_3542 1 #endif +#include + #include #include #include @@ -516,7 +518,21 @@ return std::make_unique(sockfd, reset_options); } + + bool + request_accessibility_permission() { + NSDictionary* options = @{static_cast(kAXTrustedCheckOptionPrompt): @YES}; + return !AXIsProcessTrustedWithOptions(static_cast(options)); + } + bool + has_accessibility_permission() { + NSDictionary* options = @{static_cast(kAXTrustedCheckOptionPrompt): @NO}; + // We use kAXTrustedCheckOptionPrompt == NO here, + // instead of using XIsProcessTrusted(), + // because this will update the accessibility list with sunshine current path + return AXIsProcessTrustedWithOptions(static_cast(options)); + } } // namespace platf namespace dyn {