Как настроить приложение Cocoa на клавиатуру play/pause?

Есть ли способ заставить приложение реагировать на кнопку воспроизведения/паузы на Mac?

EDIT:

Используя предлагаемый код, я получаю это сообщение консоли:

Не удалось подключить кнопку actionPressed: для целевого класса NSApplication

Почему это должно быть?

Ответ 1

Я сделал это в своем собственном приложении, создав подкласс NSApplication (и установив основной класс приложения в этот подкласс). Он ловит клавиши поиска и воспроизведения/паузы и переводит их в конкретные действия в моем делегате приложения.

Соответствующие строки:

#import <IOKit/hidsystem/ev_keymap.h>

- (void)sendEvent:(NSEvent *)event
{
    // Catch media key events
    if ([event type] == NSSystemDefined && [event subtype] == 8)
    {
        int keyCode = (([event data1] & 0xFFFF0000) >> 16);
        int keyFlags = ([event data1] & 0x0000FFFF);
        int keyState = (((keyFlags & 0xFF00) >> 8)) == 0xA;

        // Process the media key event and return
        [self mediaKeyEvent:keyCode state:keyState];
        return;
    }

    // Continue on to super
    [super sendEvent:event];
}

- (void)mediaKeyEvent:(int)key state:(BOOL)state
{
    switch (key)
    {
        // Play pressed
        case NX_KEYTYPE_PLAY:
            if (state == NO)
                [(TSAppController *)[self delegate] togglePlayPause:self];
            break;

        // Rewind
        case NX_KEYTYPE_FAST:
            if (state == YES)
                [(TSAppController *)[self delegate] seekForward:self];
            break;

        // Previous
        case NX_KEYTYPE_REWIND:
            if (state == YES)
                [(TSAppController *)[self delegate] seekBack:self];
            break;
    }
}

Edit:

Swift 4:

override func sendEvent(_ event: NSEvent)
{
    if  event.type == .systemDefined &&
        event.subtype == .screenChanged
    {
        let keyCode : Int32 = (Int32((event.data1 & 0xFFFF0000) >> 16))
        let keyFlags = (event.data1 & 0x0000FFFF)
        let keyState = ((keyFlags & 0xFF00) >> 8) == 0xA

        self.mediaKeyEvent(withKeyCode: keyCode, andState: keyState)
        return
    }

    super.sendEvent(event)
}

private func mediaKeyEvent(withKeyCode keyCode : Int32, andState state : Bool)
{
    guard let delegate = self.delegate as? AppDelegate else { return }

    switch keyCode
    {
        // Play pressed
        case NX_KEYTYPE_PLAY:
            if state == false
            {
                delegate.musicPlayerWC.handleUserPressedPlayButton()
            }
            break
        // Rewind
        case NX_KEYTYPE_FAST:
            if state == true
            {
                delegate.musicPlayerWC.handleUserPressedNextSongButton()
            }
            break

        // Previous
        case NX_KEYTYPE_REWIND:
            if state == true
            {
                delegate.musicPlayerWC.handleUserPressedPreviousSongButton()
            }

            break
        default:
            break
    }

}

Ответ 3

В случае, если кто-то ищет, есть пример кода, чтобы сделать это здесь. Такой подход позволяет вам "съедать" события, которых они не достигают в iTunes или других приложениях, поддерживающих медиа-ключ.

Но будьте осторожны, чтобы отмеченные события не были разрешены песочницей, поэтому это не будет работать в App Store. Если у кого-то есть обходное решение, я бы с удовольствием это услышал.