/* * globalaccelman_mac.cpp - manager for global hotkeys * Copyright (C) 2003 Eric Smith * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ // TODO: // - don't invoke hotkey if there is a modal dialog? // - do multi-mapping, like the x11 version #include"globalaccelman.h" #include #include #include #include typedef struct { short kchrID; Str255 KCHRname; short transtable[256]; } Ascii2KeyCodeTable; /* InitAscii2KeyCodeTable initializes the ascii to key code look up table using the currently active KCHR resource. This routine calls GetResource so it cannot be called at interrupt time.*/ OSStatus InitAscii2KeyCodeTable(Ascii2KeyCodeTable *ttable); /* ValidateAscii2KeyCodeTable verifies that the ascii to key code lookup table is synchronized with the current KCHR resource. If it is not synchronized, then the table is re-built. This routine calls GetResource so it cannot be called at interrupt time.*/ OSStatus ValidateAscii2KeyCodeTable(Ascii2KeyCodeTable *ttable,Boolean *wasChanged); /* AsciiToKeyCode looks up the ascii character in the key code look up table and returns the virtual key code for that letter. If there is no virtual key code for that letter, then the value -1 will be returned. */ short AsciiToKeyCode(Ascii2KeyCodeTable *ttable, short asciiCode); enum { kTableCountOffset = 256+2, kFirstTableOffset = 256+4, kTableSize = 128 }; OSStatus InitAscii2KeyCodeTable(Ascii2KeyCodeTable *ttable) { unsigned char *theCurrentKCHR, *ithKeyTable; short count, i, j, resID; Handle theKCHRRsrc; ResType rType; /* set up our table to all minus ones */ for (i=0;i<256; i++) ttable->transtable[i] = -1; /* find the current kchr resource ID */ ttable->kchrID = (short) GetScriptVariable(smCurrentScript, smScriptKeys); /* get the current KCHR resource */ theKCHRRsrc = GetResource('KCHR', ttable->kchrID); if (theKCHRRsrc == NULL) return resNotFound; GetResInfo(theKCHRRsrc,&resID,&rType,ttable->KCHRname); /* dereference the resource */ theCurrentKCHR = (unsigned char *) (*theKCHRRsrc); /* get the count from the resource */ count = * (short *) (theCurrentKCHR + kTableCountOffset); /* build inverse table by merging all key tables */ for (i=0; itranstable[ ithKeyTable[j] ] == -1) ttable->transtable[ ithKeyTable[j] ] = j; } } /* all done */ return noErr; } // Should probably call this at some point, in case the user has switched keyboard // layouts while we were running. OSStatus ValidateAscii2KeyCodeTable(Ascii2KeyCodeTable *ttable,Boolean *wasChanged) { short theID; theID = (short) GetScriptVariable(smCurrentScript, smScriptKeys); if (theID != ttable->kchrID) { *wasChanged=true; return InitAscii2KeyCodeTable(ttable); } else { *wasChanged=false; return noErr; } } short AsciiToKeyCode(Ascii2KeyCodeTable *ttable, short asciiCode) { if (asciiCode >= 0 && asciiCode <= 255) return ttable->transtable[asciiCode]; else return false; } // Not used. char KeyCodeToAscii(short virtualKeyCode) { unsigned long state; long keyTrans; char charCode; Ptr kchr; state = 0; kchr = (Ptr) GetScriptVariable(smCurrentScript, smKCHRCache); keyTrans = KeyTranslate(kchr, virtualKeyCode, &state); charCode = keyTrans; if (!charCode) charCode = (keyTrans>>16); return charCode; } EventHotKeyID hotKeyID = { 'QtHK', 0 }; class MacHotKey { public: MacHotKey(EventHotKeyRef _ref, int _id) { hot_key_ref = _ref; id = _id; } ~MacHotKey() { UnregisterEventHotKey(hot_key_ref); } EventHotKeyRef hot_key_ref; int id; // GlobalAccelManager unique ID for this hotkey }; // The following table is from Apple sample-code. // Apple's headers don't appear to define any constants for the virtual key // codes of special keys, but these constants are somewhat documented in the chart at // // // The constants on the chartappear to be the same values as are used in Apple's iGetKeys // sample. // . // // See also . // struct QtKeyMap { int qt_key; int mac_key; } qt_key_map[] = { { Qt::Key_Escape, 0x35 }, { Qt::Key_Tab, 0x30 }, { Qt::Key_Backtab, 0 }, { Qt::Key_Backspace, 0x33 }, { Qt::Key_Return, 0x24 }, { Qt::Key_Enter, 0x4c }, // Return & Enter are different on the Mac { Qt::Key_Insert, 0 }, { Qt::Key_Delete, 0x75 }, { Qt::Key_Pause, 0 }, { Qt::Key_Print, 0 }, { Qt::Key_SysReq, 0 }, { Qt::Key_Clear, 0x47 }, { Qt::Key_Home, 0x73 }, { Qt::Key_End, 0x77 }, { Qt::Key_Left, 0x7b }, { Qt::Key_Up, 0x7e }, { Qt::Key_Right, 0x7c }, { Qt::Key_Down, 0x7d }, { Qt::Key_Prior, 0x74 }, // Page Up { Qt::Key_Next, 0x79 }, // Page Down { Qt::Key_Shift, 0x38 }, { Qt::Key_Control, 0x3b }, { Qt::Key_Meta, 0x37 }, // Command { Qt::Key_Alt, 0x3a }, // Option { Qt::Key_CapsLock, 57 }, { Qt::Key_NumLock, 0 }, { Qt::Key_ScrollLock, 0 }, { Qt::Key_F1, 0x7a }, { Qt::Key_F2, 0x78 }, { Qt::Key_F3, 0x63 }, { Qt::Key_F4, 0x76 }, { Qt::Key_F5, 0x60 }, { Qt::Key_F6, 0x61 }, { Qt::Key_F7, 0x62 }, { Qt::Key_F8, 0x64 }, { Qt::Key_F9, 0x65 }, { Qt::Key_F10, 0x6d }, { Qt::Key_F11, 0x67 }, { Qt::Key_F12, 0x6f }, { Qt::Key_F13, 0x69 }, { Qt::Key_F14, 0x6b }, { Qt::Key_F15, 0x71 }, { Qt::Key_F16, 0 }, { Qt::Key_F17, 0 }, { Qt::Key_F18, 0 }, { Qt::Key_F19, 0 }, { Qt::Key_F20, 0 }, { Qt::Key_F21, 0 }, { Qt::Key_F22, 0 }, { Qt::Key_F23, 0 }, { Qt::Key_F24, 0 }, { Qt::Key_F25, 0 }, { Qt::Key_F26, 0 }, { Qt::Key_F27, 0 }, { Qt::Key_F28, 0 }, { Qt::Key_F29, 0 }, { Qt::Key_F30, 0 }, { Qt::Key_F31, 0 }, { Qt::Key_F32, 0 }, { Qt::Key_F33, 0 }, { Qt::Key_F34, 0 }, { Qt::Key_F35, 0 }, { Qt::Key_Super_L, 0 }, { Qt::Key_Super_R, 0 }, { Qt::Key_Menu, 0 }, { Qt::Key_Hyper_L, 0 }, { Qt::Key_Hyper_R, 0 }, { Qt::Key_Help, 0x72 }, { Qt::Key_Direction_L,0 }, { Qt::Key_Direction_R,0 }, { Qt::Key_unknown, 0 } }; class GlobalAccelManager::Private { public: Private(GlobalAccelManager *par) { id_at = 1; list.setAutoDelete(true); InitAscii2KeyCodeTable(&key_codes); hot_key_function = NewEventHandlerUPP(HotKeyHandler); EventTypeSpec type; type.eventClass = kEventClassKeyboard; type.eventKind = kEventHotKeyPressed; InstallApplicationEventHandler(hot_key_function, 1, &type, par, NULL); } ~Private() { } private: static pascal OSStatus HotKeyHandler(EventHandlerCallRef /*nextHandler*/, EventRef theEvent, void *userData); public: bool convertKeySequence(const QKeySequence &ks, UInt32 *_mod, UInt32 *_key) { int code = ks[0]; UInt32 mod = 0; if(code & Qt::META) mod |= cmdKey; if(code & Qt::SHIFT) mod |= shiftKey; if(code & Qt::CTRL) mod |= controlKey; if(code & Qt::ALT) mod |= optionKey; code &= 0xffff; UInt32 key = 0; for(int n = 0; qt_key_map[n].qt_key != Qt::Key_unknown; ++n) { if(qt_key_map[n].qt_key == code) { key = qt_key_map[n].mac_key; break; } } if (key == 0) { key = AsciiToKeyCode(&key_codes, code & 0xffff); } if(_mod) *_mod = mod; if(_key) *_key = key; return true; } int insertHotKey(int key, int mod) { EventHotKeyID hotKeyID; hotKeyID.signature = 'QtHK'; hotKeyID.id = id_at++; EventHotKeyRef hotKeyRef = 0; OSStatus foo = RegisterEventHotKey(key, mod, hotKeyID, GetApplicationEventTarget(), 0, &hotKeyRef); if (foo != 0) { fprintf(stderr, "RegisterEventHotKey(%x,%x) returned %ld\n", key, mod, foo); return 0; } MacHotKey *hk = new MacHotKey(hotKeyRef, hotKeyID.id); list.append(hk); return hk->id; } int id_at; QPtrList list; EventHandlerUPP hot_key_function; Ascii2KeyCodeTable key_codes; }; // Callback function invoked when the user hits a hot-key. pascal OSStatus GlobalAccelManager::Private::HotKeyHandler(EventHandlerCallRef /*nextHandler*/, EventRef theEvent, void *userData) { EventHotKeyID hkID; GetEventParameter(theEvent, kEventParamDirectObject, typeEventHotKeyID, NULL, sizeof(EventHotKeyID), NULL, &hkID); reinterpret_cast(userData)->triggerActivated(hkID.id); return noErr; } GlobalAccelManager::GlobalAccelManager() { d = new Private(this); } GlobalAccelManager::~GlobalAccelManager() { delete d; } int GlobalAccelManager::setAccel(const QKeySequence &ks) { UInt32 mod, key; if(!d->convertKeySequence(ks, &mod, &key)) return 0; return d->insertHotKey(key, mod); } void GlobalAccelManager::removeAccel(int id) { QPtrListIterator it(d->list); for(MacHotKey *hk = 0; (hk = it.current()); ++it) { if(hk->id == id) { d->list.removeRef(hk); return; } } }