#if UNITY_EDITOR || UNITY_ANDROID using System; using System.Linq; using System.Runtime.InteropServices; using UnityEngine.InputSystem.Layouts; using UnityEngine.InputSystem.LowLevel; using UnityEngine.InputSystem.Android.LowLevel; using UnityEngine.InputSystem.Utilities; namespace UnityEngine.InputSystem.Android.LowLevel { [StructLayout(LayoutKind.Sequential)] internal unsafe struct AndroidGameControllerState : IInputStateTypeInfo { public const int MaxAxes = 48; public const int MaxButtons = 220; internal const string kVariantGamepad = "Gamepad"; internal const string kVariantJoystick = "Joystick"; internal const uint kAxisOffset = sizeof(uint) * (uint)((MaxButtons + 31) / 32); public static FourCC kFormat = new FourCC('A', 'G', 'C', ' '); [InputControl(name = "buttonSouth", bit = (uint)AndroidKeyCode.ButtonA, variants = kVariantGamepad)] [InputControl(name = "buttonWest", bit = (uint)AndroidKeyCode.ButtonX, variants = kVariantGamepad)] [InputControl(name = "buttonNorth", bit = (uint)AndroidKeyCode.ButtonY, variants = kVariantGamepad)] [InputControl(name = "buttonEast", bit = (uint)AndroidKeyCode.ButtonB, variants = kVariantGamepad)] [InputControl(name = "leftStickPress", bit = (uint)AndroidKeyCode.ButtonThumbl, variants = kVariantGamepad)] [InputControl(name = "rightStickPress", bit = (uint)AndroidKeyCode.ButtonThumbr, variants = kVariantGamepad)] [InputControl(name = "leftShoulder", bit = (uint)AndroidKeyCode.ButtonL1, variants = kVariantGamepad)] [InputControl(name = "rightShoulder", bit = (uint)AndroidKeyCode.ButtonR1, variants = kVariantGamepad)] [InputControl(name = "start", bit = (uint)AndroidKeyCode.ButtonStart, variants = kVariantGamepad)] [InputControl(name = "select", bit = (uint)AndroidKeyCode.ButtonSelect, variants = kVariantGamepad)] public fixed uint buttons[(MaxButtons + 31) / 32]; [InputControl(name = "leftTrigger", offset = (uint)AndroidAxis.Brake * sizeof(float) + kAxisOffset, variants = kVariantGamepad)] [InputControl(name = "rightTrigger", offset = (uint)AndroidAxis.Gas * sizeof(float) + kAxisOffset, variants = kVariantGamepad)] [InputControl(name = "leftStick", variants = kVariantGamepad)] [InputControl(name = "leftStick/y", variants = kVariantGamepad, parameters = "invert")] [InputControl(name = "leftStick/up", variants = kVariantGamepad, parameters = "invert,clamp=1,clampMin=-1.0,clampMax=0.0")] [InputControl(name = "leftStick/down", variants = kVariantGamepad, parameters = "invert=false,clamp=1,clampMin=0,clampMax=1.0")] ////FIXME: state for this control is not contiguous [InputControl(name = "rightStick", offset = (uint)AndroidAxis.Z * sizeof(float) + kAxisOffset, sizeInBits = ((uint)AndroidAxis.Rz - (uint)AndroidAxis.Z) * sizeof(float) * 8, variants = kVariantGamepad)] [InputControl(name = "rightStick/x", variants = kVariantGamepad)] [InputControl(name = "rightStick/y", offset = ((uint)AndroidAxis.Rz - (uint)AndroidAxis.Z) * sizeof(float), variants = kVariantGamepad, parameters = "invert")] [InputControl(name = "rightStick/up", offset = ((uint)AndroidAxis.Rz - (uint)AndroidAxis.Z) * sizeof(float), variants = kVariantGamepad, parameters = "invert,clamp=1,clampMin=-1.0,clampMax=0.0")] [InputControl(name = "rightStick/down", offset = ((uint)AndroidAxis.Rz - (uint)AndroidAxis.Z) * sizeof(float), variants = kVariantGamepad, parameters = "invert=false,clamp=1,clampMin=0,clampMax=1.0")] public fixed float axis[MaxAxes]; public FourCC format { get { return kFormat; } } public AndroidGameControllerState WithButton(AndroidKeyCode code, bool value = true) { fixed(uint* buttonsPtr = buttons) { if (value) buttonsPtr[(int)code / 32] |= (uint)1 << ((int)code % 32); else buttonsPtr[(int)code / 32] &= ~((uint)1 << ((int)code % 32)); } return this; } public AndroidGameControllerState WithAxis(AndroidAxis axis, float value) { fixed(float* axisPtr = this.axis) { axisPtr[(int)axis] = value; } return this; } } // See https://developer.android.com/reference/android/view/InputDevice.html for input source values internal enum AndroidInputSource { Keyboard = 257, Dpad = 513, Gamepad = 1025, Touchscreen = 4098, Mouse = 8194, Stylus = 16386, Trackball = 65540, Touchpad = 1048584, Joystick = 16777232 } [Serializable] internal struct AndroidDeviceCapabilities { public string deviceDescriptor; public int productId; public int vendorId; public bool isVirtual; public AndroidAxis[] motionAxes; public AndroidInputSource inputSources; public string ToJson() { return JsonUtility.ToJson(this); } public static AndroidDeviceCapabilities FromJson(string json) { if (json == null) throw new ArgumentNullException(nameof(json)); return JsonUtility.FromJson(json); } public override string ToString() { return $"deviceDescriptor = {deviceDescriptor}, productId = {productId}, vendorId = {vendorId}, isVirtual = {isVirtual}, motionAxes = {(motionAxes == null ? "" : String.Join(",", motionAxes.Select(i => i.ToString()).ToArray()))}, inputSources = {inputSources}"; } } } namespace UnityEngine.InputSystem.Android { /// /// Most of the gamepads: /// - NVIDIA Controller v01.03/v01.04 /// - ELAN PLAYSTATION(R)3 Controller /// - My-Power CO.,LTD. PS(R) Controller Adaptor /// - Sony Interactive Entertainment Wireless /// - (Add more) /// map buttons in the following way: /// Left Stick -> AXIS_X(0) / AXIS_Y(1) /// Right Stick -> AXIS_Z (11) / AXIS_RZ(14) /// Right Thumb -> KEYCODE_BUTTON_THUMBR(107) /// Left Thumb -> KEYCODE_BUTTON_THUMBL(106) /// L1 (Left shoulder) -> KEYCODE_BUTTON_L1(102) /// R1 (Right shoulder) -> KEYCODE_BUTTON_R1(103) /// L2 (Left trigger) -> AXIS_BRAKE(23) /// R2 (Right trigger) -> AXIS_GAS(22) /// X -> KEYCODE_BUTTON_X(99) /// Y -> KEYCODE_BUTTON_Y(100) /// B -> KEYCODE_BUTTON_B(97) /// A -> KEYCODE_BUTTON_A(96) /// DPAD -> AXIS_HAT_X(15),AXIS_HAT_Y(16) or KEYCODE_DPAD_LEFT(21), KEYCODE_DPAD_RIGHT(22), KEYCODE_DPAD_UP(19), KEYCODE_DPAD_DOWN(20), /// Note: On Nvidia Shield Console, L2/R2 additionally invoke key events for AXIS_LTRIGGER, AXIS_RTRIGGER (in addition to AXIS_BRAKE, AXIS_GAS) /// If you connect gamepad to a phone for L2/R2 only AXIS_BRAKE/AXIS_GAS come. AXIS_LTRIGGER, AXIS_RTRIGGER are not invoked. /// That's why we map triggers only to AXIS_BRAKE/AXIS_GAS /// Other exotic gamepads have different mappings /// Xbox Gamepad (for ex., Microsoft X-Box One pad (Firmware 2015)) mapping (Note mapping: L2/R2/Right Stick) /// Left Stick -> AXIS_X(0) / AXIS_Y(1) /// Right Stick -> AXIS_RX (12) / AXIS_RY(13) /// Right Thumb -> KEYCODE_BUTTON_THUMBR(107) /// Left Thumb -> KEYCODE_BUTTON_THUMBL(106) /// L1 (Left shoulder) -> KEYCODE_BUTTON_L1(102) /// R1 (Right shoulder) -> KEYCODE_BUTTON_R1(103) /// L2 (Left trigger) -> AXIS_Z(11) /// R2 (Right trigger) -> AXIS_RZ(14) /// X -> KEYCODE_BUTTON_X(99) /// Y -> KEYCODE_BUTTON_Y(100) /// B -> KEYCODE_BUTTON_B(97) /// A -> KEYCODE_BUTTON_A(96) /// DPAD -> AXIS_HAT_X(15),AXIS_HAT_Y(16) /// Some gamepads on Android devices (with same Android number version) might have different mappings /// For ex., Dualshock, on NVidia Shield Console (OS 8.0) all buttons correctly map according to rules in AndroidGameControllerState /// when clicking left shoulder it will go to AndroidKeyCode.ButtonL1, rightShoulder -> AndroidKeyCode.ButtonR1, etc /// But, on Samsung Galaxy S9 (OS 8.0), the mapping is different /// when clicking left shoulder it will go to AndroidKeyCode.ButtonY, rightShoulder -> AndroidKeyCode.ButtonZ, etc /// So even though Android version is 8.0 in both cases, Dualshock will only correctly work on NVidia Shield Console /// It's obvious that this depends on the driver and not Android OS, thus we can only assume Samsung in this case doesn't properly support Dualshock in their drivers /// While we can do custom mapping for Samsung, we can never now when will they try to update the driver for Dualshock or some other gamepad /// [InputControlLayout(stateType = typeof(AndroidGameControllerState), variants = AndroidGameControllerState.kVariantGamepad)] [Scripting.Preserve] public class AndroidGamepad : Gamepad { } [InputControlLayout(stateType = typeof(AndroidGameControllerState), variants = AndroidGameControllerState.kVariantJoystick)] [Scripting.Preserve] public class AndroidJoystick : Joystick { } } #endif // UNITY_EDITOR || UNITY_ANDROID