2015-06-21 15:33:46 +00:00
|
|
|
/*
|
|
|
|
Simple DirectMedia Layer
|
2019-01-05 06:01:14 +00:00
|
|
|
Copyright (C) 1997-2019 Sam Lantinga <slouken@libsdl.org>
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
#include "../../SDL_internal.h"
|
|
|
|
|
|
|
|
/* This is the iOS implementation of the SDL joystick API */
|
2015-09-21 02:08:36 +00:00
|
|
|
#include "SDL_sysjoystick_c.h"
|
|
|
|
|
|
|
|
/* needed for SDL_IPHONE_MAX_GFORCE macro */
|
|
|
|
#include "SDL_config_iphoneos.h"
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2017-09-22 15:30:52 +00:00
|
|
|
#include "SDL_assert.h"
|
2016-08-26 19:46:29 +00:00
|
|
|
#include "SDL_events.h"
|
2015-06-21 15:33:46 +00:00
|
|
|
#include "SDL_joystick.h"
|
|
|
|
#include "SDL_hints.h"
|
|
|
|
#include "SDL_stdinc.h"
|
|
|
|
#include "../SDL_sysjoystick.h"
|
|
|
|
#include "../SDL_joystick_c.h"
|
2017-09-22 15:30:52 +00:00
|
|
|
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !SDL_EVENTS_DISABLED
|
|
|
|
#include "../../events/SDL_events_c.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !TARGET_OS_TV
|
2015-06-21 15:33:46 +00:00
|
|
|
#import <CoreMotion/CoreMotion.h>
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
#import <GameController/GameController.h>
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
static id connectObserver = nil;
|
|
|
|
static id disconnectObserver = nil;
|
2019-06-06 15:20:53 +00:00
|
|
|
|
|
|
|
#include <Availability.h>
|
|
|
|
#include <objc/message.h>
|
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
/* remove compilation warnings for strict builds by defining these selectors, even though
|
|
|
|
* they are only ever used indirectly through objc_msgSend
|
|
|
|
*/
|
2019-06-06 15:20:53 +00:00
|
|
|
@interface GCExtendedGamepad (SDL)
|
2019-06-14 20:56:52 +00:00
|
|
|
#if (__IPHONE_OS_VERSION_MAX_ALLOWED < 121000) || (__APPLETV_OS_VERSION_MAX_ALLOWED < 121000) || (__MAC_OS_VERSION_MAX_ALLOWED < 1401000)
|
|
|
|
@property (nonatomic, readonly, nullable) GCControllerButtonInput *leftThumbstickButton;
|
|
|
|
@property (nonatomic, readonly, nullable) GCControllerButtonInput *rightThumbstickButton;
|
|
|
|
#endif
|
|
|
|
#if (__IPHONE_OS_VERSION_MAX_ALLOWED < 130000) || (__APPLETV_OS_VERSION_MAX_ALLOWED < 130000) || (__MAC_OS_VERSION_MAX_ALLOWED < 1500000)
|
2019-06-06 15:20:53 +00:00
|
|
|
@property (nonatomic, readonly) GCControllerButtonInput *buttonMenu;
|
|
|
|
@property (nonatomic, readonly, nullable) GCControllerButtonInput *buttonOptions;
|
|
|
|
#endif
|
2019-06-14 20:56:52 +00:00
|
|
|
@end
|
|
|
|
@interface GCMicroGamepad (SDL)
|
|
|
|
#if (__IPHONE_OS_VERSION_MAX_ALLOWED < 130000) || (__APPLETV_OS_VERSION_MAX_ALLOWED < 130000) || (__MAC_OS_VERSION_MAX_ALLOWED < 1500000)
|
|
|
|
@property (nonatomic, readonly) GCControllerButtonInput *buttonMenu;
|
2019-06-06 15:20:53 +00:00
|
|
|
#endif
|
|
|
|
@end
|
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2015-09-21 02:08:36 +00:00
|
|
|
static const char *accelerometerName = "iOS Accelerometer";
|
2015-06-21 15:33:46 +00:00
|
|
|
static CMMotionManager *motionManager = nil;
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
static SDL_JoystickDeviceItem *deviceList = NULL;
|
|
|
|
|
2015-06-21 15:33:46 +00:00
|
|
|
static int numjoysticks = 0;
|
2018-02-07 00:43:31 +00:00
|
|
|
int SDL_AppleTVRemoteOpenedAsJoystick = 0;
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
static SDL_JoystickDeviceItem *
|
|
|
|
GetDeviceForIndex(int device_index)
|
|
|
|
{
|
|
|
|
SDL_JoystickDeviceItem *device = deviceList;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
while (i < device_index) {
|
|
|
|
if (device == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
device = device->next;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddMFIJoystickDevice(SDL_JoystickDeviceItem *device, GCController *controller)
|
2015-09-21 02:08:36 +00:00
|
|
|
{
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
2018-03-09 00:32:22 +00:00
|
|
|
const Uint16 VENDOR_APPLE = 0x05AC;
|
2019-06-06 15:20:53 +00:00
|
|
|
const Uint16 VENDOR_MICROSOFT = 0x045e;
|
|
|
|
const Uint16 VENDOR_SONY = 0x054C;
|
2018-03-08 02:09:58 +00:00
|
|
|
Uint16 *guid16 = (Uint16 *)device->guid.data;
|
2018-03-09 00:32:22 +00:00
|
|
|
Uint16 vendor = 0;
|
|
|
|
Uint16 product = 0;
|
|
|
|
Uint8 subtype = 0;
|
2018-03-08 02:09:58 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
const char *name = NULL;
|
|
|
|
/* Explicitly retain the controller because SDL_JoystickDeviceItem is a
|
|
|
|
* struct, and ARC doesn't work with structs. */
|
|
|
|
device->controller = (__bridge GCController *) CFBridgingRetain(controller);
|
|
|
|
|
|
|
|
if (controller.vendorName) {
|
|
|
|
name = controller.vendorName.UTF8String;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!name) {
|
|
|
|
name = "MFi Gamepad";
|
|
|
|
}
|
|
|
|
|
|
|
|
device->name = SDL_strdup(name);
|
|
|
|
|
|
|
|
if (controller.extendedGamepad) {
|
2019-06-14 20:56:52 +00:00
|
|
|
GCExtendedGamepad *gamepad = controller.extendedGamepad;
|
|
|
|
int nbuttons = 0;
|
|
|
|
|
|
|
|
/* These buttons are part of the original MFi spec */
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_A);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_B);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_X);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_Y);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_LEFTSHOULDER);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_RIGHTSHOULDER);
|
|
|
|
nbuttons += 6;
|
|
|
|
|
|
|
|
/* These buttons are available on some newer controllers */
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
#pragma clang diagnostic ignored "-Wunguarded-availability-new"
|
|
|
|
if ([gamepad respondsToSelector:@selector(leftThumbstickButton)] && gamepad.leftThumbstickButton) {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_LEFTSTICK);
|
|
|
|
++nbuttons;
|
|
|
|
}
|
|
|
|
if ([gamepad respondsToSelector:@selector(rightThumbstickButton)] && gamepad.rightThumbstickButton) {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_RIGHTSTICK);
|
|
|
|
++nbuttons;
|
2019-06-06 15:20:53 +00:00
|
|
|
}
|
2019-06-14 20:56:52 +00:00
|
|
|
if ([gamepad respondsToSelector:@selector(buttonOptions)] && gamepad.buttonOptions) {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_BACK);
|
|
|
|
++nbuttons;
|
|
|
|
}
|
|
|
|
if ([gamepad respondsToSelector:@selector(buttonMenu)] && gamepad.buttonMenu) {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_START);
|
|
|
|
++nbuttons;
|
|
|
|
} else {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_START);
|
|
|
|
++nbuttons;
|
|
|
|
device->uses_pause_handler = SDL_TRUE;
|
|
|
|
}
|
|
|
|
#pragma clang diagnostic pop
|
2019-06-06 15:20:53 +00:00
|
|
|
|
|
|
|
if ([controller.vendorName containsString: @"Xbox"]) {
|
|
|
|
vendor = VENDOR_MICROSOFT;
|
2019-06-14 20:56:52 +00:00
|
|
|
product = 0x02E0; /* Assume Xbox One S BLE Controller unless/until GCController flows VID/PID */
|
2019-06-06 15:20:53 +00:00
|
|
|
} else if ([controller.vendorName containsString: @"DUALSHOCK"]) {
|
|
|
|
vendor = VENDOR_SONY;
|
2019-06-14 20:56:52 +00:00
|
|
|
product = 0x09CC; /* Assume DS4 Slim unless/until GCController flows VID/PID */
|
2019-06-06 15:20:53 +00:00
|
|
|
} else {
|
|
|
|
vendor = VENDOR_APPLE;
|
|
|
|
product = 1;
|
|
|
|
subtype = 1;
|
|
|
|
}
|
2019-06-14 20:56:52 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
device->naxes = 6; /* 2 thumbsticks and 2 triggers */
|
|
|
|
device->nhats = 1; /* d-pad */
|
2019-06-06 15:20:53 +00:00
|
|
|
device->nbuttons = nbuttons;
|
2019-06-14 20:56:52 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
} else if (controller.gamepad) {
|
2019-06-14 20:56:52 +00:00
|
|
|
int nbuttons = 0;
|
|
|
|
|
|
|
|
/* These buttons are part of the original MFi spec */
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_A);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_B);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_X);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_Y);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_LEFTSHOULDER);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_RIGHTSHOULDER);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_START);
|
|
|
|
nbuttons += 7;
|
|
|
|
device->uses_pause_handler = SDL_TRUE;
|
|
|
|
|
2018-03-09 00:32:22 +00:00
|
|
|
vendor = VENDOR_APPLE;
|
|
|
|
product = 2;
|
|
|
|
subtype = 2;
|
2015-09-21 02:08:36 +00:00
|
|
|
device->naxes = 0; /* no traditional analog inputs */
|
|
|
|
device->nhats = 1; /* d-pad */
|
2019-06-14 20:56:52 +00:00
|
|
|
device->nbuttons = nbuttons;
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2016-09-14 01:18:06 +00:00
|
|
|
#if TARGET_OS_TV
|
|
|
|
else if (controller.microGamepad) {
|
2019-06-14 20:56:52 +00:00
|
|
|
GCMicroGamepad *gamepad = controller.microGamepad;
|
|
|
|
int nbuttons = 0;
|
|
|
|
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_A);
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_B); /* Button X on microGamepad */
|
|
|
|
nbuttons += 2;
|
|
|
|
|
|
|
|
if ([gamepad respondsToSelector:@selector(buttonMenu)] && gamepad.buttonMenu) {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_START);
|
|
|
|
++nbuttons;
|
|
|
|
} else {
|
|
|
|
device->button_mask |= (1 << SDL_CONTROLLER_BUTTON_START);
|
|
|
|
++nbuttons;
|
|
|
|
device->uses_pause_handler = SDL_TRUE;
|
|
|
|
}
|
|
|
|
|
2018-03-09 00:32:22 +00:00
|
|
|
vendor = VENDOR_APPLE;
|
|
|
|
product = 3;
|
|
|
|
subtype = 3;
|
2016-09-14 01:18:06 +00:00
|
|
|
device->naxes = 2; /* treat the touch surface as two axes */
|
|
|
|
device->nhats = 0; /* apparently the touch surface-as-dpad is buggy */
|
2019-06-14 20:56:52 +00:00
|
|
|
device->nbuttons = nbuttons;
|
2016-09-17 04:31:07 +00:00
|
|
|
|
2016-10-08 06:40:44 +00:00
|
|
|
controller.microGamepad.allowsRotation = SDL_GetHintBoolean(SDL_HINT_APPLE_TV_REMOTE_ALLOW_ROTATION, SDL_FALSE);
|
2016-09-14 01:18:06 +00:00
|
|
|
}
|
|
|
|
#endif /* TARGET_OS_TV */
|
2015-12-11 20:41:59 +00:00
|
|
|
|
2018-03-08 02:09:58 +00:00
|
|
|
/* We only need 16 bits for each of these; space them out to fill 128. */
|
|
|
|
/* Byteswap so devices get same GUID on little/big endian platforms. */
|
2018-08-09 23:00:17 +00:00
|
|
|
*guid16++ = SDL_SwapLE16(SDL_HARDWARE_BUS_BLUETOOTH);
|
2018-03-08 02:09:58 +00:00
|
|
|
*guid16++ = 0;
|
2018-03-09 00:32:22 +00:00
|
|
|
*guid16++ = SDL_SwapLE16(vendor);
|
|
|
|
*guid16++ = 0;
|
|
|
|
*guid16++ = SDL_SwapLE16(product);
|
|
|
|
*guid16++ = 0;
|
2018-03-08 02:09:58 +00:00
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
*guid16++ = SDL_SwapLE16(device->button_mask);
|
|
|
|
|
|
|
|
if (subtype != 0) {
|
|
|
|
/* Note that this is an MFI controller and what subtype it is */
|
|
|
|
device->guid.data[14] = 'm';
|
|
|
|
device->guid.data[15] = subtype;
|
|
|
|
}
|
2018-03-08 02:09:58 +00:00
|
|
|
|
2015-12-11 20:41:59 +00:00
|
|
|
/* This will be set when the first button press of the controller is
|
|
|
|
* detected. */
|
|
|
|
controller.playerIndex = -1;
|
2016-09-14 01:18:06 +00:00
|
|
|
|
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddJoystickDevice(GCController *controller, SDL_bool accelerometer)
|
2015-09-21 02:08:36 +00:00
|
|
|
{
|
|
|
|
SDL_JoystickDeviceItem *device = deviceList;
|
|
|
|
|
2018-02-06 23:03:38 +00:00
|
|
|
#if TARGET_OS_TV
|
|
|
|
if (!SDL_GetHintBoolean(SDL_HINT_TV_REMOTE_AS_JOYSTICK, SDL_TRUE)) {
|
|
|
|
/* Ignore devices that aren't actually controllers (e.g. remotes), they'll be handled as keyboard input */
|
|
|
|
if (controller && !controller.extendedGamepad && !controller.gamepad && controller.microGamepad) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
while (device != NULL) {
|
|
|
|
if (device->controller == controller) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
device = device->next;
|
|
|
|
}
|
|
|
|
|
2018-02-06 23:03:38 +00:00
|
|
|
device = (SDL_JoystickDeviceItem *) SDL_calloc(1, sizeof(SDL_JoystickDeviceItem));
|
2015-09-21 02:08:36 +00:00
|
|
|
if (device == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
device->accelerometer = accelerometer;
|
2018-08-09 23:00:17 +00:00
|
|
|
device->instance_id = SDL_GetNextJoystickInstanceID();
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
if (accelerometer) {
|
2016-09-14 01:18:06 +00:00
|
|
|
#if TARGET_OS_TV
|
|
|
|
SDL_free(device);
|
|
|
|
return;
|
|
|
|
#else
|
2015-09-21 02:08:36 +00:00
|
|
|
device->name = SDL_strdup(accelerometerName);
|
|
|
|
device->naxes = 3; /* Device acceleration in the x, y, and z axes. */
|
|
|
|
device->nhats = 0;
|
|
|
|
device->nbuttons = 0;
|
|
|
|
|
|
|
|
/* Use the accelerometer name as a GUID. */
|
|
|
|
SDL_memcpy(&device->guid.data, device->name, SDL_min(sizeof(SDL_JoystickGUID), SDL_strlen(device->name)));
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
} else if (controller) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddMFIJoystickDevice(device, controller);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (deviceList == NULL) {
|
|
|
|
deviceList = device;
|
|
|
|
} else {
|
|
|
|
SDL_JoystickDeviceItem *lastdevice = deviceList;
|
|
|
|
while (lastdevice->next != NULL) {
|
|
|
|
lastdevice = lastdevice->next;
|
|
|
|
}
|
|
|
|
lastdevice->next = device;
|
|
|
|
}
|
|
|
|
|
|
|
|
++numjoysticks;
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
SDL_PrivateJoystickAdded(device->instance_id);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
2015-09-25 18:17:20 +00:00
|
|
|
static SDL_JoystickDeviceItem *
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_RemoveJoystickDevice(SDL_JoystickDeviceItem *device)
|
2015-09-21 02:08:36 +00:00
|
|
|
{
|
|
|
|
SDL_JoystickDeviceItem *prev = NULL;
|
|
|
|
SDL_JoystickDeviceItem *next = NULL;
|
|
|
|
SDL_JoystickDeviceItem *item = deviceList;
|
|
|
|
|
|
|
|
if (device == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
next = device->next;
|
|
|
|
|
|
|
|
while (item != NULL) {
|
|
|
|
if (item == device) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
prev = item;
|
|
|
|
item = item->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unlink the device item from the device list. */
|
|
|
|
if (prev) {
|
|
|
|
prev->next = device->next;
|
|
|
|
} else if (device == deviceList) {
|
|
|
|
deviceList = device->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (device->joystick) {
|
|
|
|
device->joystick->hwdata = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
@autoreleasepool {
|
|
|
|
if (device->controller) {
|
|
|
|
/* The controller was explicitly retained in the struct, so it
|
|
|
|
* should be explicitly released before freeing the struct. */
|
|
|
|
GCController *controller = CFBridgingRelease((__bridge CFTypeRef)(device->controller));
|
|
|
|
controller.controllerPausedHandler = nil;
|
|
|
|
device->controller = nil;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
|
|
|
|
|
|
|
--numjoysticks;
|
|
|
|
|
2017-09-22 15:30:52 +00:00
|
|
|
SDL_PrivateJoystickRemoved(device->instance_id);
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2015-09-21 19:19:37 +00:00
|
|
|
SDL_free(device->name);
|
|
|
|
SDL_free(device);
|
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
return next;
|
|
|
|
}
|
2015-06-21 15:33:46 +00:00
|
|
|
|
2016-09-17 04:31:07 +00:00
|
|
|
#if TARGET_OS_TV
|
2017-08-14 13:28:21 +00:00
|
|
|
static void SDLCALL
|
2016-09-17 04:31:07 +00:00
|
|
|
SDL_AppleTVRemoteRotationHintChanged(void *udata, const char *name, const char *oldValue, const char *newValue)
|
|
|
|
{
|
|
|
|
BOOL allowRotation = newValue != NULL && *newValue != '0';
|
|
|
|
|
|
|
|
@autoreleasepool {
|
|
|
|
for (GCController *controller in [GCController controllers]) {
|
|
|
|
if (controller.microGamepad) {
|
|
|
|
controller.microGamepad.allowsRotation = allowRotation;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* TARGET_OS_TV */
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static int
|
|
|
|
IOS_JoystickInit(void)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
@autoreleasepool {
|
|
|
|
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
|
|
|
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2016-10-08 06:40:44 +00:00
|
|
|
if (SDL_GetHintBoolean(SDL_HINT_ACCELEROMETER_AS_JOYSTICK, SDL_TRUE)) {
|
2015-09-21 02:08:36 +00:00
|
|
|
/* Default behavior, accelerometer as joystick */
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddJoystickDevice(nil, SDL_TRUE);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
/* GameController.framework was added in iOS 7. */
|
|
|
|
if (![GCController class]) {
|
2018-08-09 23:00:17 +00:00
|
|
|
return 0;
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (GCController *controller in [GCController controllers]) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddJoystickDevice(controller, SDL_FALSE);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
2016-09-17 04:31:07 +00:00
|
|
|
#if TARGET_OS_TV
|
|
|
|
SDL_AddHintCallback(SDL_HINT_APPLE_TV_REMOTE_ALLOW_ROTATION,
|
|
|
|
SDL_AppleTVRemoteRotationHintChanged, NULL);
|
|
|
|
#endif /* TARGET_OS_TV */
|
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
connectObserver = [center addObserverForName:GCControllerDidConnectNotification
|
|
|
|
object:nil
|
|
|
|
queue:nil
|
|
|
|
usingBlock:^(NSNotification *note) {
|
|
|
|
GCController *controller = note.object;
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AddJoystickDevice(controller, SDL_FALSE);
|
2015-09-21 02:08:36 +00:00
|
|
|
}];
|
|
|
|
|
|
|
|
disconnectObserver = [center addObserverForName:GCControllerDidDisconnectNotification
|
|
|
|
object:nil
|
|
|
|
queue:nil
|
|
|
|
usingBlock:^(NSNotification *note) {
|
|
|
|
GCController *controller = note.object;
|
|
|
|
SDL_JoystickDeviceItem *device = deviceList;
|
|
|
|
while (device != NULL) {
|
|
|
|
if (device->controller == controller) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_RemoveJoystickDevice(device);
|
2015-09-21 02:08:36 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
device = device->next;
|
|
|
|
}
|
|
|
|
}];
|
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
return 0;
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static int
|
|
|
|
IOS_JoystickGetCount(void)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
|
|
|
return numjoysticks;
|
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static void
|
|
|
|
IOS_JoystickDetect(void)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static const char *
|
|
|
|
IOS_JoystickGetDeviceName(int device_index)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_JoystickDeviceItem *device = GetDeviceForIndex(device_index);
|
|
|
|
return device ? device->name : "Unknown";
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:53:14 +00:00
|
|
|
static int
|
|
|
|
IOS_JoystickGetDevicePlayerIndex(int device_index)
|
|
|
|
{
|
2019-06-14 20:56:52 +00:00
|
|
|
SDL_JoystickDeviceItem *device = GetDeviceForIndex(device_index);
|
|
|
|
return device ? (int)device->controller.playerIndex : -1;
|
2018-10-25 23:53:14 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static SDL_JoystickGUID
|
|
|
|
IOS_JoystickGetDeviceGUID( int device_index )
|
|
|
|
{
|
|
|
|
SDL_JoystickDeviceItem *device = GetDeviceForIndex(device_index);
|
|
|
|
SDL_JoystickGUID guid;
|
|
|
|
if (device) {
|
|
|
|
guid = device->guid;
|
|
|
|
} else {
|
|
|
|
SDL_zero(guid);
|
|
|
|
}
|
|
|
|
return guid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SDL_JoystickID
|
|
|
|
IOS_JoystickGetDeviceInstanceID(int device_index)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_JoystickDeviceItem *device = GetDeviceForIndex(device_index);
|
2018-08-09 23:00:17 +00:00
|
|
|
return device ? device->instance_id : -1;
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static int
|
|
|
|
IOS_JoystickOpen(SDL_Joystick * joystick, int device_index)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_JoystickDeviceItem *device = GetDeviceForIndex(device_index);
|
|
|
|
if (device == NULL) {
|
|
|
|
return SDL_SetError("Could not open Joystick: no hardware device for the specified index");
|
|
|
|
}
|
|
|
|
|
|
|
|
joystick->hwdata = device;
|
|
|
|
joystick->instance_id = device->instance_id;
|
|
|
|
|
|
|
|
joystick->naxes = device->naxes;
|
|
|
|
joystick->nhats = device->nhats;
|
|
|
|
joystick->nbuttons = device->nbuttons;
|
2015-06-21 15:33:46 +00:00
|
|
|
joystick->nballs = 0;
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
device->joystick = joystick;
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
@autoreleasepool {
|
2015-09-21 02:08:36 +00:00
|
|
|
if (device->accelerometer) {
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2015-09-21 02:08:36 +00:00
|
|
|
if (motionManager == nil) {
|
|
|
|
motionManager = [[CMMotionManager alloc] init];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Shorter times between updates can significantly increase CPU usage. */
|
|
|
|
motionManager.accelerometerUpdateInterval = 0.1;
|
|
|
|
[motionManager startAccelerometerUpdates];
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
} else {
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
2019-06-14 20:56:52 +00:00
|
|
|
if (device->uses_pause_handler) {
|
|
|
|
GCController *controller = device->controller;
|
|
|
|
controller.controllerPausedHandler = ^(GCController *c) {
|
|
|
|
if (joystick->hwdata) {
|
|
|
|
++joystick->hwdata->num_pause_presses;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2015-09-21 02:08:36 +00:00
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
}
|
2018-02-07 00:43:31 +00:00
|
|
|
if (device->remote) {
|
|
|
|
++SDL_AppleTVRemoteOpenedAsJoystick;
|
|
|
|
}
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-09-25 18:17:20 +00:00
|
|
|
static void
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AccelerometerUpdate(SDL_Joystick * joystick)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2015-06-21 15:33:46 +00:00
|
|
|
const float maxgforce = SDL_IPHONE_MAX_GFORCE;
|
|
|
|
const SInt16 maxsint16 = 0x7FFF;
|
|
|
|
CMAcceleration accel;
|
|
|
|
|
|
|
|
@autoreleasepool {
|
2015-09-21 02:08:36 +00:00
|
|
|
if (!motionManager.isAccelerometerActive) {
|
2015-06-21 15:33:46 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
accel = motionManager.accelerometerData.acceleration;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Convert accelerometer data from floating point to Sint16, which is what
|
|
|
|
the joystick system expects.
|
|
|
|
|
|
|
|
To do the conversion, the data is first clamped onto the interval
|
|
|
|
[-SDL_IPHONE_MAX_G_FORCE, SDL_IPHONE_MAX_G_FORCE], then the data is multiplied
|
|
|
|
by MAX_SINT16 so that it is mapped to the full range of an Sint16.
|
|
|
|
|
|
|
|
You can customize the clamped range of this function by modifying the
|
|
|
|
SDL_IPHONE_MAX_GFORCE macro in SDL_config_iphoneos.h.
|
|
|
|
|
|
|
|
Once converted to Sint16, the accelerometer data no longer has coherent
|
|
|
|
units. You can convert the data back to units of g-force by multiplying
|
|
|
|
it in your application's code by SDL_IPHONE_MAX_GFORCE / 0x7FFF.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* clamp the data */
|
|
|
|
accel.x = SDL_min(SDL_max(accel.x, -maxgforce), maxgforce);
|
|
|
|
accel.y = SDL_min(SDL_max(accel.y, -maxgforce), maxgforce);
|
|
|
|
accel.z = SDL_min(SDL_max(accel.z, -maxgforce), maxgforce);
|
|
|
|
|
|
|
|
/* pass in data mapped to range of SInt16 */
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_PrivateJoystickAxis(joystick, 0, (accel.x / maxgforce) * maxsint16);
|
2015-06-21 15:33:46 +00:00
|
|
|
SDL_PrivateJoystickAxis(joystick, 1, -(accel.y / maxgforce) * maxsint16);
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_PrivateJoystickAxis(joystick, 2, (accel.z / maxgforce) * maxsint16);
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
static Uint8
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_MFIJoystickHatStateForDPad(GCControllerDirectionPad *dpad)
|
2015-09-21 02:08:36 +00:00
|
|
|
{
|
|
|
|
Uint8 hat = 0;
|
|
|
|
|
|
|
|
if (dpad.up.isPressed) {
|
|
|
|
hat |= SDL_HAT_UP;
|
|
|
|
} else if (dpad.down.isPressed) {
|
|
|
|
hat |= SDL_HAT_DOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dpad.left.isPressed) {
|
|
|
|
hat |= SDL_HAT_LEFT;
|
|
|
|
} else if (dpad.right.isPressed) {
|
|
|
|
hat |= SDL_HAT_RIGHT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hat == 0) {
|
|
|
|
return SDL_HAT_CENTERED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hat;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_MFIJoystickUpdate(SDL_Joystick * joystick)
|
2015-09-21 02:08:36 +00:00
|
|
|
{
|
2016-09-14 01:18:06 +00:00
|
|
|
#if SDL_JOYSTICK_MFI
|
2015-09-21 02:08:36 +00:00
|
|
|
@autoreleasepool {
|
|
|
|
GCController *controller = joystick->hwdata->controller;
|
|
|
|
Uint8 hatstate = SDL_HAT_CENTERED;
|
|
|
|
int i;
|
2015-12-11 20:41:59 +00:00
|
|
|
int updateplayerindex = 0;
|
2019-06-14 20:56:52 +00:00
|
|
|
int pause_button_index = 0;
|
2015-09-21 02:08:36 +00:00
|
|
|
|
|
|
|
if (controller.extendedGamepad) {
|
|
|
|
GCExtendedGamepad *gamepad = controller.extendedGamepad;
|
|
|
|
|
|
|
|
/* Axis order matches the XInput Windows mappings. */
|
2015-12-11 20:41:59 +00:00
|
|
|
Sint16 axes[] = {
|
|
|
|
(Sint16) (gamepad.leftThumbstick.xAxis.value * 32767),
|
|
|
|
(Sint16) (gamepad.leftThumbstick.yAxis.value * -32767),
|
|
|
|
(Sint16) ((gamepad.leftTrigger.value * 65535) - 32768),
|
|
|
|
(Sint16) (gamepad.rightThumbstick.xAxis.value * 32767),
|
|
|
|
(Sint16) (gamepad.rightThumbstick.yAxis.value * -32767),
|
|
|
|
(Sint16) ((gamepad.rightTrigger.value * 65535) - 32768),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Button order matches the XInput Windows mappings. */
|
2019-06-06 15:20:53 +00:00
|
|
|
Uint8 buttons[joystick->nbuttons];
|
2019-06-14 20:56:52 +00:00
|
|
|
int button_count = 0;
|
|
|
|
|
|
|
|
/* These buttons are part of the original MFi spec */
|
|
|
|
buttons[button_count++] = gamepad.buttonA.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonB.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonX.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonY.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.leftShoulder.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.rightShoulder.isPressed;
|
|
|
|
|
|
|
|
/* These buttons are available on some newer controllers */
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
#pragma clang diagnostic ignored "-Wunguarded-availability-new"
|
|
|
|
if (joystick->hwdata->button_mask & (1 << SDL_CONTROLLER_BUTTON_LEFTSTICK)) {
|
|
|
|
buttons[button_count++] = gamepad.leftThumbstickButton.isPressed;
|
2019-06-06 15:20:53 +00:00
|
|
|
}
|
2019-06-14 20:56:52 +00:00
|
|
|
if (joystick->hwdata->button_mask & (1 << SDL_CONTROLLER_BUTTON_RIGHTSTICK)) {
|
|
|
|
buttons[button_count++] = gamepad.rightThumbstickButton.isPressed;
|
2019-06-06 15:20:53 +00:00
|
|
|
}
|
2019-06-14 20:56:52 +00:00
|
|
|
if (joystick->hwdata->button_mask & (1 << SDL_CONTROLLER_BUTTON_BACK)) {
|
|
|
|
buttons[button_count++] = gamepad.buttonOptions.isPressed;
|
|
|
|
}
|
|
|
|
/* This must be the last button, so we can optionally handle it with pause_button_index below */
|
|
|
|
if (joystick->hwdata->button_mask & (1 << SDL_CONTROLLER_BUTTON_START)) {
|
|
|
|
if (joystick->hwdata->uses_pause_handler) {
|
|
|
|
pause_button_index = button_count;
|
2019-07-13 01:28:43 +00:00
|
|
|
buttons[button_count++] = joystick->delayed_guide_button;
|
2019-06-14 20:56:52 +00:00
|
|
|
} else {
|
|
|
|
buttons[button_count++] = gamepad.buttonMenu.isPressed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#pragma clang diagnostic pop
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
hatstate = IOS_MFIJoystickHatStateForDPad(gamepad.dpad);
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2015-12-11 20:41:59 +00:00
|
|
|
for (i = 0; i < SDL_arraysize(axes); i++) {
|
|
|
|
/* The triggers (axes 2 and 5) are resting at -32768 but SDL
|
|
|
|
* initializes its values to 0. We only want to make sure the
|
|
|
|
* player index is up to date if the user actually moves an axis. */
|
|
|
|
if ((i != 2 && i != 5) || axes[i] != -32768) {
|
2016-12-23 01:43:00 +00:00
|
|
|
updateplayerindex |= (joystick->axes[i].value != axes[i]);
|
2015-12-11 20:41:59 +00:00
|
|
|
}
|
|
|
|
SDL_PrivateJoystickAxis(joystick, i, axes[i]);
|
|
|
|
}
|
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
for (i = 0; i < button_count; i++) {
|
2015-12-11 20:41:59 +00:00
|
|
|
updateplayerindex |= (joystick->buttons[i] != buttons[i]);
|
|
|
|
SDL_PrivateJoystickButton(joystick, i, buttons[i]);
|
|
|
|
}
|
2015-09-21 02:08:36 +00:00
|
|
|
} else if (controller.gamepad) {
|
|
|
|
GCGamepad *gamepad = controller.gamepad;
|
|
|
|
|
2015-12-11 20:41:59 +00:00
|
|
|
/* Button order matches the XInput Windows mappings. */
|
2019-06-14 20:56:52 +00:00
|
|
|
Uint8 buttons[joystick->nbuttons];
|
|
|
|
int button_count = 0;
|
|
|
|
buttons[button_count++] = gamepad.buttonA.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonB.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonX.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonY.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.leftShoulder.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.rightShoulder.isPressed;
|
|
|
|
pause_button_index = button_count;
|
2019-07-13 01:28:43 +00:00
|
|
|
buttons[button_count++] = joystick->delayed_guide_button;
|
2015-12-11 20:41:59 +00:00
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
hatstate = IOS_MFIJoystickHatStateForDPad(gamepad.dpad);
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
for (i = 0; i < button_count; i++) {
|
2015-12-11 20:41:59 +00:00
|
|
|
updateplayerindex |= (joystick->buttons[i] != buttons[i]);
|
|
|
|
SDL_PrivateJoystickButton(joystick, i, buttons[i]);
|
|
|
|
}
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2016-09-14 01:18:06 +00:00
|
|
|
#if TARGET_OS_TV
|
|
|
|
else if (controller.microGamepad) {
|
|
|
|
GCMicroGamepad *gamepad = controller.microGamepad;
|
|
|
|
|
|
|
|
Sint16 axes[] = {
|
|
|
|
(Sint16) (gamepad.dpad.xAxis.value * 32767),
|
|
|
|
(Sint16) (gamepad.dpad.yAxis.value * -32767),
|
|
|
|
};
|
|
|
|
|
|
|
|
for (i = 0; i < SDL_arraysize(axes); i++) {
|
2017-02-03 00:56:02 +00:00
|
|
|
updateplayerindex |= (joystick->axes[i].value != axes[i]);
|
2016-09-14 01:18:06 +00:00
|
|
|
SDL_PrivateJoystickAxis(joystick, i, axes[i]);
|
|
|
|
}
|
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
Uint8 buttons[joystick->nbuttons];
|
|
|
|
int button_count = 0;
|
|
|
|
buttons[button_count++] = gamepad.buttonA.isPressed;
|
|
|
|
buttons[button_count++] = gamepad.buttonX.isPressed;
|
|
|
|
#pragma clang diagnostic push
|
|
|
|
#pragma clang diagnostic ignored "-Wunguarded-availability-new"
|
|
|
|
/* This must be the last button, so we can optionally handle it with pause_button_index below */
|
|
|
|
if (joystick->hwdata->button_mask & (1 << SDL_CONTROLLER_BUTTON_START)) {
|
|
|
|
if (joystick->hwdata->uses_pause_handler) {
|
|
|
|
pause_button_index = button_count;
|
2019-07-13 01:28:43 +00:00
|
|
|
buttons[button_count++] = joystick->delayed_guide_button;
|
2019-06-14 20:56:52 +00:00
|
|
|
} else {
|
|
|
|
buttons[button_count++] = gamepad.buttonMenu.isPressed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#pragma clang diagnostic pop
|
2016-09-14 01:18:06 +00:00
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
for (i = 0; i < button_count; i++) {
|
2016-09-14 01:18:06 +00:00
|
|
|
updateplayerindex |= (joystick->buttons[i] != buttons[i]);
|
|
|
|
SDL_PrivateJoystickButton(joystick, i, buttons[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2015-12-11 20:41:59 +00:00
|
|
|
if (joystick->nhats > 0) {
|
|
|
|
updateplayerindex |= (joystick->hats[0] != hatstate);
|
|
|
|
SDL_PrivateJoystickHat(joystick, 0, hatstate);
|
|
|
|
}
|
2015-09-21 02:08:36 +00:00
|
|
|
|
2019-06-14 20:56:52 +00:00
|
|
|
if (joystick->hwdata->uses_pause_handler) {
|
|
|
|
for (i = 0; i < joystick->hwdata->num_pause_presses; i++) {
|
|
|
|
SDL_PrivateJoystickButton(joystick, pause_button_index, SDL_PRESSED);
|
|
|
|
SDL_PrivateJoystickButton(joystick, pause_button_index, SDL_RELEASED);
|
|
|
|
updateplayerindex = YES;
|
|
|
|
}
|
|
|
|
joystick->hwdata->num_pause_presses = 0;
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2015-12-11 20:41:59 +00:00
|
|
|
|
|
|
|
if (updateplayerindex && controller.playerIndex == -1) {
|
|
|
|
BOOL usedPlayerIndexSlots[4] = {NO, NO, NO, NO};
|
|
|
|
|
|
|
|
/* Find the player index of all other connected controllers. */
|
|
|
|
for (GCController *c in [GCController controllers]) {
|
|
|
|
if (c != controller && c.playerIndex >= 0) {
|
|
|
|
usedPlayerIndexSlots[c.playerIndex] = YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set this controller's player index to the first unused index.
|
|
|
|
* FIXME: This logic isn't great... but SDL doesn't expose this
|
|
|
|
* concept in its external API, so we don't have much to go on. */
|
|
|
|
for (i = 0; i < SDL_arraysize(usedPlayerIndexSlots); i++) {
|
|
|
|
if (!usedPlayerIndexSlots[i]) {
|
|
|
|
controller.playerIndex = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static int
|
|
|
|
IOS_JoystickRumble(SDL_Joystick * joystick, Uint16 low_frequency_rumble, Uint16 high_frequency_rumble, Uint32 duration_ms)
|
|
|
|
{
|
|
|
|
return SDL_Unsupported();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
IOS_JoystickUpdate(SDL_Joystick * joystick)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_JoystickDeviceItem *device = joystick->hwdata;
|
|
|
|
|
|
|
|
if (device == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
2017-09-22 15:30:52 +00:00
|
|
|
|
2015-09-21 02:08:36 +00:00
|
|
|
if (device->accelerometer) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_AccelerometerUpdate(joystick);
|
2015-09-21 02:08:36 +00:00
|
|
|
} else if (device->controller) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_MFIJoystickUpdate(joystick);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static void
|
|
|
|
IOS_JoystickClose(SDL_Joystick * joystick)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2015-09-21 02:08:36 +00:00
|
|
|
SDL_JoystickDeviceItem *device = joystick->hwdata;
|
|
|
|
|
|
|
|
if (device == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
device->joystick = NULL;
|
|
|
|
|
2015-06-21 15:33:46 +00:00
|
|
|
@autoreleasepool {
|
2015-09-21 02:08:36 +00:00
|
|
|
if (device->accelerometer) {
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2015-09-21 02:08:36 +00:00
|
|
|
[motionManager stopAccelerometerUpdates];
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
} else if (device->controller) {
|
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
GCController *controller = device->controller;
|
|
|
|
controller.controllerPausedHandler = nil;
|
2015-12-11 20:41:59 +00:00
|
|
|
controller.playerIndex = -1;
|
2015-09-21 02:08:36 +00:00
|
|
|
#endif
|
|
|
|
}
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
2018-02-07 00:43:31 +00:00
|
|
|
if (device->remote) {
|
|
|
|
--SDL_AppleTVRemoteOpenedAsJoystick;
|
|
|
|
}
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
static void
|
|
|
|
IOS_JoystickQuit(void)
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
|
|
|
@autoreleasepool {
|
2015-09-21 02:08:36 +00:00
|
|
|
#ifdef SDL_JOYSTICK_MFI
|
|
|
|
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
|
|
|
|
|
|
|
if (connectObserver) {
|
|
|
|
[center removeObserver:connectObserver name:GCControllerDidConnectNotification object:nil];
|
|
|
|
connectObserver = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (disconnectObserver) {
|
|
|
|
[center removeObserver:disconnectObserver name:GCControllerDidDisconnectNotification object:nil];
|
|
|
|
disconnectObserver = nil;
|
|
|
|
}
|
2016-09-17 04:31:07 +00:00
|
|
|
|
|
|
|
#if TARGET_OS_TV
|
|
|
|
SDL_DelHintCallback(SDL_HINT_APPLE_TV_REMOTE_ALLOW_ROTATION,
|
|
|
|
SDL_AppleTVRemoteRotationHintChanged, NULL);
|
|
|
|
#endif /* TARGET_OS_TV */
|
2015-09-21 02:08:36 +00:00
|
|
|
#endif /* SDL_JOYSTICK_MFI */
|
|
|
|
|
|
|
|
while (deviceList != NULL) {
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_RemoveJoystickDevice(deviceList);
|
2015-09-21 02:08:36 +00:00
|
|
|
}
|
|
|
|
|
2016-09-14 01:18:06 +00:00
|
|
|
#if !TARGET_OS_TV
|
2015-06-21 15:33:46 +00:00
|
|
|
motionManager = nil;
|
2016-09-14 01:18:06 +00:00
|
|
|
#endif /* !TARGET_OS_TV */
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
numjoysticks = 0;
|
|
|
|
}
|
|
|
|
|
2018-08-09 23:00:17 +00:00
|
|
|
SDL_JoystickDriver SDL_IOS_JoystickDriver =
|
2015-06-21 15:33:46 +00:00
|
|
|
{
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_JoystickInit,
|
|
|
|
IOS_JoystickGetCount,
|
|
|
|
IOS_JoystickDetect,
|
|
|
|
IOS_JoystickGetDeviceName,
|
2018-10-25 23:53:14 +00:00
|
|
|
IOS_JoystickGetDevicePlayerIndex,
|
2018-08-09 23:00:17 +00:00
|
|
|
IOS_JoystickGetDeviceGUID,
|
|
|
|
IOS_JoystickGetDeviceInstanceID,
|
|
|
|
IOS_JoystickOpen,
|
|
|
|
IOS_JoystickRumble,
|
|
|
|
IOS_JoystickUpdate,
|
|
|
|
IOS_JoystickClose,
|
|
|
|
IOS_JoystickQuit,
|
|
|
|
};
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|