/* ============================================================================== This file is part of the JUCE library - "Jules' Utility Class Extensions" Copyright 2004-7 by Raw Material Software ltd. ------------------------------------------------------------------------------ JUCE can be redistributed and/or modified under the terms of the GNU General Public License, as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. JUCE 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 General Public License for more details. You should have received a copy of the GNU General Public License along with JUCE; if not, visit www.gnu.org/licenses or write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ------------------------------------------------------------------------------ If you'd like to release a closed-source product which uses JUCE, commercial licenses are also available: visit www.rawmaterialsoftware.com/juce for more information. ============================================================================== */ // (This file gets included by juce_mac_NativeCode.mm, rather than being // compiled on its own). #ifdef JUCE_INCLUDED_FILE //============================================================================== AppleRemoteDevice::AppleRemoteDevice() : device (0), queue (0), remoteId (0) { } AppleRemoteDevice::~AppleRemoteDevice() { stop(); } static io_object_t getAppleRemoteDevice() throw() { CFMutableDictionaryRef dict = IOServiceMatching ("AppleIRController"); io_iterator_t iter = 0; io_object_t iod = 0; if (IOServiceGetMatchingServices (kIOMasterPortDefault, dict, &iter) == kIOReturnSuccess && iter != 0) { iod = IOIteratorNext (iter); } IOObjectRelease (iter); return iod; } static bool createAppleRemoteInterface (io_object_t iod, void** device) throw() { jassert (*device == 0); io_name_t classname; if (IOObjectGetClass (iod, classname) == kIOReturnSuccess) { IOCFPlugInInterface** cfPlugInInterface = 0; SInt32 score = 0; if (IOCreatePlugInInterfaceForService (iod, kIOHIDDeviceUserClientTypeID, kIOCFPlugInInterfaceID, &cfPlugInInterface, &score) == kIOReturnSuccess) { HRESULT hr = (*cfPlugInInterface)->QueryInterface (cfPlugInInterface, CFUUIDGetUUIDBytes (kIOHIDDeviceInterfaceID), device); (void) hr; (*cfPlugInInterface)->Release (cfPlugInInterface); } } return *device != 0; } bool AppleRemoteDevice::start (const bool inExclusiveMode) throw() { if (queue != 0) return true; stop(); bool result = false; io_object_t iod = getAppleRemoteDevice(); if (iod != 0) { if (createAppleRemoteInterface (iod, &device) && open (inExclusiveMode)) result = true; else stop(); IOObjectRelease (iod); } return result; } void AppleRemoteDevice::stop() throw() { if (queue != 0) { (*(IOHIDQueueInterface**) queue)->stop ((IOHIDQueueInterface**) queue); (*(IOHIDQueueInterface**) queue)->dispose ((IOHIDQueueInterface**) queue); (*(IOHIDQueueInterface**) queue)->Release ((IOHIDQueueInterface**) queue); queue = 0; } if (device != 0) { (*(IOHIDDeviceInterface**) device)->close ((IOHIDDeviceInterface**) device); (*(IOHIDDeviceInterface**) device)->Release ((IOHIDDeviceInterface**) device); device = 0; } } bool AppleRemoteDevice::isActive() const throw() { return queue != 0; } static void appleRemoteQueueCallback (void* const target, const IOReturn result, void*, void*) { if (result == kIOReturnSuccess) ((AppleRemoteDevice*) target)->handleCallbackInternal(); } bool AppleRemoteDevice::open (const bool openInExclusiveMode) throw() { #if ! MACOS_10_2_OR_EARLIER Array cookies; CFArrayRef elements; IOHIDDeviceInterface122** const device122 = (IOHIDDeviceInterface122**) device; if ((*device122)->copyMatchingElements (device122, 0, &elements) != kIOReturnSuccess) return false; for (int i = 0; i < CFArrayGetCount (elements); ++i) { CFDictionaryRef element = (CFDictionaryRef) CFArrayGetValueAtIndex (elements, i); // get the cookie CFTypeRef object = CFDictionaryGetValue (element, CFSTR (kIOHIDElementCookieKey)); if (object == 0 || CFGetTypeID (object) != CFNumberGetTypeID()) continue; long number; if (! CFNumberGetValue ((CFNumberRef) object, kCFNumberLongType, &number)) continue; cookies.add ((int) number); } CFRelease (elements); if ((*(IOHIDDeviceInterface**) device) ->open ((IOHIDDeviceInterface**) device, openInExclusiveMode ? kIOHIDOptionsTypeSeizeDevice : kIOHIDOptionsTypeNone) == KERN_SUCCESS) { queue = (*(IOHIDDeviceInterface**) device)->allocQueue ((IOHIDDeviceInterface**) device); if (queue != 0) { (*(IOHIDQueueInterface**) queue)->create ((IOHIDQueueInterface**) queue, 0, 12); for (int i = 0; i < cookies.size(); ++i) { IOHIDElementCookie cookie = (IOHIDElementCookie) cookies.getUnchecked(i); (*(IOHIDQueueInterface**) queue)->addElement ((IOHIDQueueInterface**) queue, cookie, 0); } CFRunLoopSourceRef eventSource; if ((*(IOHIDQueueInterface**) queue) ->createAsyncEventSource ((IOHIDQueueInterface**) queue, &eventSource) == KERN_SUCCESS) { if ((*(IOHIDQueueInterface**) queue)->setEventCallout ((IOHIDQueueInterface**) queue, appleRemoteQueueCallback, this, 0) == KERN_SUCCESS) { CFRunLoopAddSource (CFRunLoopGetCurrent(), eventSource, kCFRunLoopDefaultMode); (*(IOHIDQueueInterface**) queue)->start ((IOHIDQueueInterface**) queue); return true; } } } } #endif return false; } void AppleRemoteDevice::handleCallbackInternal() { #if ! MACOS_10_2_OR_EARLIER int totalValues = 0; AbsoluteTime nullTime = { 0, 0 }; char cookies [12]; int numCookies = 0; while (numCookies < numElementsInArray (cookies)) { IOHIDEventStruct e; if ((*(IOHIDQueueInterface**) queue)->getNextEvent ((IOHIDQueueInterface**) queue, &e, nullTime, 0) != kIOReturnSuccess) break; if ((int) e.elementCookie == 19) { remoteId = e.value; buttonPressed (switched, false); } else { totalValues += e.value; cookies [numCookies++] = (char) (pointer_sized_int) e.elementCookie; } } cookies [numCookies++] = 0; //DBG (String::toHexString ((uint8*) cookies, numCookies, 1) + " " + String (totalValues)); static const char buttonPatterns[] = { 0x1f, 0x14, 0x12, 0x1f, 0x14, 0x12, 0, 0x1f, 0x15, 0x12, 0x1f, 0x15, 0x12, 0, 0x1f, 0x1d, 0x1c, 0x12, 0, 0x1f, 0x1e, 0x1c, 0x12, 0, 0x1f, 0x16, 0x12, 0x1f, 0x16, 0x12, 0, 0x1f, 0x17, 0x12, 0x1f, 0x17, 0x12, 0, 0x1f, 0x12, 0x04, 0x02, 0, 0x1f, 0x12, 0x03, 0x02, 0, 0x1f, 0x12, 0x1f, 0x12, 0, 0x23, 0x1f, 0x12, 0x23, 0x1f, 0x12, 0, 19, 0 }; int buttonNum = (int) menuButton; int i = 0; while (i < numElementsInArray (buttonPatterns)) { if (strcmp (cookies, buttonPatterns + i) == 0) { buttonPressed ((ButtonType) buttonNum, totalValues > 0); break; } i += strlen (buttonPatterns + i) + 1; ++buttonNum; } #endif } #endif