SDL for Mac - only enable global event tap when actually necessary (app has focus and has requested relative mouse mode or has asked for a mouse grab). in other situations the event tap impacts system performance and battery life with no benefit.

This commit is contained in:
Sam Lantinga 2016-11-26 10:26:22 -08:00
parent ff56c7b300
commit 354a8f276e
3 changed files with 36 additions and 12 deletions

View file

@ -26,6 +26,7 @@
#include "SDL_cocoamouse.h" #include "SDL_cocoamouse.h"
extern void Cocoa_InitMouseEventTap(SDL_MouseData *driverdata); extern void Cocoa_InitMouseEventTap(SDL_MouseData *driverdata);
extern void Cocoa_EnableMouseEventTap(SDL_MouseData *driverdata, SDL_bool enabled);
extern void Cocoa_QuitMouseEventTap(SDL_MouseData *driverdata); extern void Cocoa_QuitMouseEventTap(SDL_MouseData *driverdata);
#endif /* _SDL_cocoamousetap_h */ #endif /* _SDL_cocoamousetap_h */

View file

@ -142,15 +142,12 @@ Cocoa_MouseTapThread(void *data)
{ {
SDL_MouseEventTapData *tapdata = (SDL_MouseEventTapData*)data; SDL_MouseEventTapData *tapdata = (SDL_MouseEventTapData*)data;
/* Create a tap. */ /* Tap was created on main thread but we own it now. */
CFMachPortRef eventTap = CGEventTapCreate(kCGSessionEventTap, kCGHeadInsertEventTap, CFMachPortRef eventTap = tapdata->tap;
kCGEventTapOptionDefault, allGrabbedEventsMask,
&Cocoa_MouseTapCallback, tapdata);
if (eventTap) { if (eventTap) {
/* Try to create a runloop source we can schedule. */ /* Try to create a runloop source we can schedule. */
CFRunLoopSourceRef runloopSource = CFMachPortCreateRunLoopSource(kCFAllocatorDefault, eventTap, 0); CFRunLoopSourceRef runloopSource = CFMachPortCreateRunLoopSource(kCFAllocatorDefault, eventTap, 0);
if (runloopSource) { if (runloopSource) {
tapdata->tap = eventTap;
tapdata->runloopSource = runloopSource; tapdata->runloopSource = runloopSource;
} else { } else {
CFRelease(eventTap); CFRelease(eventTap);
@ -202,16 +199,31 @@ Cocoa_InitMouseEventTap(SDL_MouseData* driverdata)
tapdata->runloopStartedSemaphore = SDL_CreateSemaphore(0); tapdata->runloopStartedSemaphore = SDL_CreateSemaphore(0);
if (tapdata->runloopStartedSemaphore) { if (tapdata->runloopStartedSemaphore) {
tapdata->tap = CGEventTapCreate(kCGSessionEventTap, kCGHeadInsertEventTap,
kCGEventTapOptionDefault, allGrabbedEventsMask,
&Cocoa_MouseTapCallback, tapdata);
if (tapdata->tap) {
tapdata->thread = SDL_CreateThreadInternal(&Cocoa_MouseTapThread, "Event Tap Loop", 512 * 1024, tapdata); tapdata->thread = SDL_CreateThreadInternal(&Cocoa_MouseTapThread, "Event Tap Loop", 512 * 1024, tapdata);
if (!tapdata->thread) { if (tapdata->thread) {
/* Success - early out. Ownership transferred to thread. */
return;
}
CFRelease(tapdata->tap);
}
SDL_DestroySemaphore(tapdata->runloopStartedSemaphore); SDL_DestroySemaphore(tapdata->runloopStartedSemaphore);
} }
}
if (!tapdata->thread) {
SDL_free(driverdata->tapdata); SDL_free(driverdata->tapdata);
driverdata->tapdata = NULL; driverdata->tapdata = NULL;
} }
void
Cocoa_EnableMouseEventTap(SDL_MouseData *driverdata, SDL_bool enabled)
{
SDL_MouseEventTapData *tapdata = (SDL_MouseEventTapData*)driverdata->tapdata;
if (tapdata && tapdata->tap)
{
CGEventTapEnable(tapdata->tap, enabled);
}
} }
void void
@ -245,6 +257,11 @@ Cocoa_InitMouseEventTap(SDL_MouseData *unused)
{ {
} }
void
Cocoa_EnableMouseEventTap(SDL_MouseData *driverdata, SDL_bool enabled)
{
}
void void
Cocoa_QuitMouseEventTap(SDL_MouseData *driverdata) Cocoa_QuitMouseEventTap(SDL_MouseData *driverdata)
{ {

View file

@ -38,6 +38,7 @@
#include "SDL_cocoavideo.h" #include "SDL_cocoavideo.h"
#include "SDL_cocoashape.h" #include "SDL_cocoashape.h"
#include "SDL_cocoamouse.h" #include "SDL_cocoamouse.h"
#include "SDL_cocoamousetap.h"
#include "SDL_cocoaopengl.h" #include "SDL_cocoaopengl.h"
#include "SDL_assert.h" #include "SDL_assert.h"
@ -1634,8 +1635,13 @@ Cocoa_GetWindowGammaRamp(_THIS, SDL_Window * window, Uint16 * ramp)
void void
Cocoa_SetWindowGrab(_THIS, SDL_Window * window, SDL_bool grabbed) Cocoa_SetWindowGrab(_THIS, SDL_Window * window, SDL_bool grabbed)
{ {
/* Move the cursor to the nearest point in the window */ SDL_Mouse *mouse = SDL_GetMouse();
SDL_WindowData *data = (SDL_WindowData *) window->driverdata; SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
/* Enable or disable the event tap as necessary */
Cocoa_EnableMouseEventTap(mouse->driverdata, grabbed);
/* Move the cursor to the nearest point in the window */
if (grabbed && data && ![data->listener isMoving]) { if (grabbed && data && ![data->listener isMoving]) {
int x, y; int x, y;
CGPoint cgpoint; CGPoint cgpoint;