How to consume CAPS-LOCK event on MacOS

2019-08-18 00:50发布

问题:

I am trying to remap CAPS-LOCK.

I can successfully detect CAPS-LOCK key-down/up events using CGEventTap.

However, I am unable to consume them. (OTOH I am able to consume ordinary key-down/up events).

The below code excerpt illustrates this:

@implementation Intercept

- (BOOL) tapEvents
{
    _eventTap = CGEventTapCreate( kCGHIDEventTap, // kCGSessionEventTap,
                                  kCGHeadInsertEventTap,
                                  kCGEventTapOptionDefault,
                                       CGEventMaskBit( NSEventTypeKeyDown )
                                     | CGEventMaskBit( NSEventTypeFlagsChanged )
                                     | CGEventMaskBit( NSEventTypeSystemDefined ),
                                  (CGEventTapCallBack) tapCallback,
                                  (__bridge void *)(self));
    CGEventTapEnable( _eventTap, YES );
}


CGEventRef tapCallback(
                        CGEventTapProxy proxy,
                        CGEventType     type,
                        CGEventRef      event,
                        Intercept*     listener
                        )
{
    return [listener interceptEvent:event];
}


- (CGEventRef)interceptEvent:(CGEventRef)cgEvent
{    
    NSEvent* event = [NSEvent eventWithCGEvent:cgEvent];

    /*
     CAPSLOCK sends:
       (1) NSEventTypeSystemDefined
       (2) NSEventTypeFlagsChanged
     */

    switch( event.type ) {
        // if I return NULL here, it DOESN'T eat CAPS-LOCK
        case NSEventTypeFlagsChanged:
        case NSEventTypeSystemDefined:
            return NULL;

        // if I return NULL here, it DOES eat keystrokes
        case NSEventTypeKeyDown:
            return NULL;
    }

    CGEventRef ev = [event CGEvent];
    CFRetain(ev);
    return ev;
}

How come this one key escapes the rules that bind the rest?

Do I have to drop down to IOHID layer?

REF: Simulate/Toggle CAPS LOCK programatically in OS X

Alternatively, might it make sense to inject antidote CAPS-LOCK events that will cancel the effect of the real ones?