2de9c0b907
Fixes #209. Negating short.MinValue would cause a wrap around back to short.MinValue. This resulted in joystick inputs like 0.8, 0.9, -1.00031.
475 lines
16 KiB
C#
475 lines
16 KiB
C#
#region License
|
|
//
|
|
// XInputJoystick.cs
|
|
//
|
|
// Author:
|
|
// Stefanos A. <stapostol@gmail.com>
|
|
//
|
|
// Copyright (c) 2006-2013 Stefanos Apostolopoulos
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
// THE SOFTWARE.
|
|
//
|
|
#endregion
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using OpenTK.Input;
|
|
using System.Runtime.InteropServices;
|
|
using System.Security;
|
|
using System.Diagnostics;
|
|
|
|
namespace OpenTK.Platform.Windows
|
|
{
|
|
class XInputJoystick : IJoystickDriver2, IDisposable
|
|
{
|
|
// All XInput devices use the same Guid
|
|
// (only one GamePadConfiguration entry required)
|
|
static readonly Guid guid =
|
|
new Guid("78696e70757400000000000000000000"); // equiv. to "xinput"
|
|
|
|
XInput xinput = new XInput();
|
|
|
|
#region IJoystickDriver2 Members
|
|
|
|
public JoystickState GetState(int index)
|
|
{
|
|
XInputState xstate;
|
|
XInputErrorCode error = xinput.GetState((XInputUserIndex)index, out xstate);
|
|
|
|
JoystickState state = new JoystickState();
|
|
if (error == XInputErrorCode.Success)
|
|
{
|
|
state.SetIsConnected(true);
|
|
|
|
state.SetAxis(JoystickAxis.Axis0, (short)xstate.GamePad.ThumbLX);
|
|
state.SetAxis(JoystickAxis.Axis1, (short)Math.Min(short.MaxValue, -xstate.GamePad.ThumbLY));
|
|
state.SetAxis(JoystickAxis.Axis2, (short)Common.HidHelper.ScaleValue(xstate.GamePad.LeftTrigger, 0, byte.MaxValue, short.MinValue, short.MaxValue));
|
|
state.SetAxis(JoystickAxis.Axis3, (short)xstate.GamePad.ThumbRX);
|
|
state.SetAxis(JoystickAxis.Axis4, (short)Math.Min(short.MaxValue, -xstate.GamePad.ThumbRY));
|
|
state.SetAxis(JoystickAxis.Axis5, (short)Common.HidHelper.ScaleValue(xstate.GamePad.RightTrigger, 0, byte.MaxValue, short.MinValue, short.MaxValue));
|
|
|
|
state.SetButton(JoystickButton.Button0, (xstate.GamePad.Buttons & XInputButtons.A) != 0);
|
|
state.SetButton(JoystickButton.Button1, (xstate.GamePad.Buttons & XInputButtons.B) != 0);
|
|
state.SetButton(JoystickButton.Button2, (xstate.GamePad.Buttons & XInputButtons.X) != 0);
|
|
state.SetButton(JoystickButton.Button3, (xstate.GamePad.Buttons & XInputButtons.Y) != 0);
|
|
state.SetButton(JoystickButton.Button4, (xstate.GamePad.Buttons & XInputButtons.LeftShoulder) != 0);
|
|
state.SetButton(JoystickButton.Button5, (xstate.GamePad.Buttons & XInputButtons.RightShoulder) != 0);
|
|
state.SetButton(JoystickButton.Button6, (xstate.GamePad.Buttons & XInputButtons.Back) != 0);
|
|
state.SetButton(JoystickButton.Button7, (xstate.GamePad.Buttons & XInputButtons.Start) != 0);
|
|
state.SetButton(JoystickButton.Button8, (xstate.GamePad.Buttons & XInputButtons.LeftThumb) != 0);
|
|
state.SetButton(JoystickButton.Button9, (xstate.GamePad.Buttons & XInputButtons.RightThumb) != 0);
|
|
state.SetButton(JoystickButton.Button10, (xstate.GamePad.Buttons & XInputButtons.Guide) != 0);
|
|
|
|
state.SetHat(JoystickHat.Hat0, new JoystickHatState(TranslateHat(xstate.GamePad.Buttons)));
|
|
}
|
|
|
|
return state;
|
|
}
|
|
|
|
private HatPosition TranslateHat(XInputButtons buttons)
|
|
{
|
|
XInputButtons dir = 0;
|
|
|
|
dir =XInputButtons.DPadUp | XInputButtons.DPadLeft;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.UpLeft;
|
|
dir = XInputButtons.DPadUp | XInputButtons.DPadRight;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.UpRight;
|
|
dir = XInputButtons.DPadDown | XInputButtons.DPadLeft;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.DownLeft;
|
|
dir = XInputButtons.DPadDown | XInputButtons.DPadRight;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.DownRight;
|
|
|
|
dir = XInputButtons.DPadUp;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.Up;
|
|
dir = XInputButtons.DPadRight;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.Right;
|
|
dir = XInputButtons.DPadDown;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.Down;
|
|
dir = XInputButtons.DPadLeft;
|
|
if ((buttons & dir) == dir)
|
|
return HatPosition.Left;
|
|
|
|
return HatPosition.Centered;
|
|
}
|
|
|
|
public JoystickCapabilities GetCapabilities(int index)
|
|
{
|
|
XInputDeviceCapabilities xcaps;
|
|
XInputErrorCode error = xinput.GetCapabilities(
|
|
(XInputUserIndex)index,
|
|
XInputCapabilitiesFlags.Default,
|
|
out xcaps);
|
|
|
|
if (error == XInputErrorCode.Success)
|
|
{
|
|
//GamePadType type = TranslateSubType(xcaps.SubType);
|
|
int buttons = TranslateButtons(xcaps.GamePad.Buttons);
|
|
int axes = TranslateAxes(ref xcaps.GamePad);
|
|
|
|
return new JoystickCapabilities(axes, buttons, 0, true);
|
|
}
|
|
return new JoystickCapabilities();
|
|
}
|
|
|
|
public string GetName(int index)
|
|
{
|
|
return String.Empty;
|
|
}
|
|
|
|
public Guid GetGuid(int index)
|
|
{
|
|
return guid;
|
|
}
|
|
|
|
public bool SetVibration(int index, float left, float right)
|
|
{
|
|
left = MathHelper.Clamp(left, 0.0f, 1.0f);
|
|
right = MathHelper.Clamp(right, 0.0f, 1.0f);
|
|
|
|
XInputVibration vibration = new XInputVibration(
|
|
(ushort)(left * UInt16.MaxValue),
|
|
(ushort)(right * UInt16.MaxValue));
|
|
|
|
return xinput.SetState((XInputUserIndex)index, ref vibration) == XInputErrorCode.Success;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Private Members
|
|
|
|
int TranslateAxes(ref XInputGamePad pad)
|
|
{
|
|
int count = 0;
|
|
count += pad.ThumbLX != 0 ? 1 : 0;
|
|
count += pad.ThumbLY != 0 ? 1 : 0;
|
|
count += pad.ThumbRX != 0 ? 1 : 0;
|
|
count += pad.ThumbRY != 0 ? 1 : 0;
|
|
count += pad.LeftTrigger != 0 ? 1 : 0;
|
|
count += pad.RightTrigger != 0 ? 1 : 0;
|
|
return count;
|
|
}
|
|
|
|
int NumberOfSetBits(int i)
|
|
{
|
|
i = i - ((i >> 1) & 0x55555555);
|
|
i = (i & 0x33333333) + ((i >> 2) & 0x33333333);
|
|
return (((i + (i >> 4)) & 0x0F0F0F0F) * 0x01010101) >> 24;
|
|
}
|
|
|
|
int TranslateButtons(XInputButtons xbuttons)
|
|
{
|
|
return NumberOfSetBits((int)xbuttons);
|
|
}
|
|
|
|
#if false
|
|
// Todo: Implement JoystickType enumeration
|
|
GamePadType TranslateSubType(XInputDeviceSubType xtype)
|
|
{
|
|
switch (xtype)
|
|
{
|
|
case XInputDeviceSubType.ArcadePad: return GamePadType.ArcadePad;
|
|
case XInputDeviceSubType.ArcadeStick: return GamePadType.ArcadeStick;
|
|
case XInputDeviceSubType.DancePad: return GamePadType.DancePad;
|
|
case XInputDeviceSubType.DrumKit: return GamePadType.DrumKit;
|
|
case XInputDeviceSubType.FlightStick: return GamePadType.FlightStick;
|
|
case XInputDeviceSubType.GamePad: return GamePadType.GamePad;
|
|
case XInputDeviceSubType.Guitar: return GamePadType.Guitar;
|
|
case XInputDeviceSubType.GuitarAlternate: return GamePadType.AlternateGuitar;
|
|
case XInputDeviceSubType.GuitarBass: return GamePadType.BassGuitar;
|
|
case XInputDeviceSubType.Wheel: return GamePadType.Wheel;
|
|
case XInputDeviceSubType.Unknown:
|
|
default:
|
|
return GamePadType.Unknown;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
enum XInputErrorCode
|
|
{
|
|
Success = 0,
|
|
DeviceNotConnected
|
|
}
|
|
|
|
enum XInputDeviceType : byte
|
|
{
|
|
GamePad
|
|
}
|
|
|
|
enum XInputDeviceSubType : byte
|
|
{
|
|
Unknown = 0,
|
|
GamePad = 1,
|
|
Wheel = 2,
|
|
ArcadeStick = 3,
|
|
FlightStick = 4,
|
|
DancePad = 5,
|
|
Guitar = 6,
|
|
GuitarAlternate = 7,
|
|
DrumKit = 8,
|
|
GuitarBass = 0xb,
|
|
ArcadePad = 0x13
|
|
}
|
|
|
|
enum XInputCapabilities
|
|
{
|
|
ForceFeedback = 0x0001,
|
|
Wireless = 0x0002,
|
|
Voice = 0x0004,
|
|
PluginModules = 0x0008,
|
|
NoNavigation = 0x0010,
|
|
}
|
|
|
|
enum XInputButtons : ushort
|
|
{
|
|
DPadUp = 0x0001,
|
|
DPadDown = 0x0002,
|
|
DPadLeft = 0x0004,
|
|
DPadRight = 0x0008,
|
|
Start = 0x0010,
|
|
Back = 0x0020,
|
|
LeftThumb = 0x0040,
|
|
RightThumb = 0x0080,
|
|
LeftShoulder = 0x0100,
|
|
RightShoulder = 0x0200,
|
|
Guide = 0x0400, // Undocumented, requires XInputGetStateEx + XINPUT_1_3.dll or higher
|
|
A = 0x1000,
|
|
B = 0x2000,
|
|
X = 0x4000,
|
|
Y = 0x8000
|
|
}
|
|
|
|
[Flags]
|
|
enum XInputCapabilitiesFlags
|
|
{
|
|
Default = 0,
|
|
GamePadOnly = 1
|
|
}
|
|
|
|
enum XInputBatteryType : byte
|
|
{
|
|
Disconnected = 0x00,
|
|
Wired = 0x01,
|
|
Alkaline = 0x02,
|
|
NiMH = 0x03,
|
|
Unknown = 0xff
|
|
}
|
|
|
|
enum XInputBatteryLevel : byte
|
|
{
|
|
Empty = 0x00,
|
|
Low = 0x01,
|
|
Medium = 0x02,
|
|
Full = 0x03
|
|
}
|
|
|
|
enum XInputUserIndex
|
|
{
|
|
First = 0,
|
|
Second,
|
|
Third,
|
|
Fourth,
|
|
Any = 0xff
|
|
}
|
|
|
|
#pragma warning disable 0649 // field is never assigned
|
|
|
|
struct XInputThresholds
|
|
{
|
|
public const int LeftThumbDeadzone = 7849;
|
|
public const int RightThumbDeadzone = 8689;
|
|
public const int TriggerThreshold = 30;
|
|
}
|
|
|
|
struct XInputGamePad
|
|
{
|
|
public XInputButtons Buttons;
|
|
public byte LeftTrigger;
|
|
public byte RightTrigger;
|
|
public short ThumbLX;
|
|
public short ThumbLY;
|
|
public short ThumbRX;
|
|
public short ThumbRY;
|
|
}
|
|
|
|
struct XInputState
|
|
{
|
|
public int PacketNumber;
|
|
public XInputGamePad GamePad;
|
|
}
|
|
|
|
struct XInputVibration
|
|
{
|
|
public ushort LeftMotorSpeed;
|
|
public ushort RightMotorSpeed;
|
|
|
|
public XInputVibration(ushort left, ushort right)
|
|
{
|
|
LeftMotorSpeed = left;
|
|
RightMotorSpeed = right;
|
|
}
|
|
}
|
|
|
|
struct XInputDeviceCapabilities
|
|
{
|
|
public XInputDeviceType Type;
|
|
public XInputDeviceSubType SubType;
|
|
public short Flags;
|
|
public XInputGamePad GamePad;
|
|
public XInputVibration Vibration;
|
|
}
|
|
|
|
struct XInputBatteryInformation
|
|
{
|
|
public XInputBatteryType Type;
|
|
public XInputBatteryLevel Level;
|
|
}
|
|
|
|
class XInput : IDisposable
|
|
{
|
|
IntPtr dll;
|
|
|
|
internal XInput()
|
|
{
|
|
// Try to load the newest XInput***.dll installed on the system
|
|
// The delegates below will be loaded dynamically from that dll
|
|
dll = Functions.LoadLibrary("XINPUT1_4");
|
|
if (dll == IntPtr.Zero)
|
|
dll = Functions.LoadLibrary("XINPUT1_3");
|
|
if (dll == IntPtr.Zero)
|
|
dll = Functions.LoadLibrary("XINPUT1_2");
|
|
if (dll == IntPtr.Zero)
|
|
dll = Functions.LoadLibrary("XINPUT1_1");
|
|
if (dll == IntPtr.Zero)
|
|
dll = Functions.LoadLibrary("XINPUT9_1_0");
|
|
if (dll == IntPtr.Zero)
|
|
throw new NotSupportedException("XInput was not found on this platform");
|
|
|
|
// Load the entry points we are interested in from that dll
|
|
GetCapabilities = (XInputGetCapabilities)Load("XInputGetCapabilities", typeof(XInputGetCapabilities));
|
|
GetState =
|
|
// undocumented XInputGetStateEx with support for the "Guide" button (requires XINPUT_1_3+)
|
|
(XInputGetState)Load("XInputGetStateEx", typeof(XInputGetState)) ??
|
|
// documented XInputGetState (no support for the "Guide" button)
|
|
(XInputGetState)Load("XInputGetState", typeof(XInputGetState));
|
|
SetState = (XInputSetState)Load("XInputSetState", typeof(XInputSetState));
|
|
}
|
|
|
|
#region Private Members
|
|
|
|
Delegate Load(string name, Type type)
|
|
{
|
|
IntPtr pfunc = Functions.GetProcAddress(dll, name);
|
|
if (pfunc != IntPtr.Zero)
|
|
return Marshal.GetDelegateForFunctionPointer(pfunc, type);
|
|
return null;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Internal Members
|
|
|
|
internal XInputGetCapabilities GetCapabilities;
|
|
internal XInputGetState GetState;
|
|
internal XInputSetState SetState;
|
|
|
|
[SuppressUnmanagedCodeSecurity]
|
|
internal delegate XInputErrorCode XInputGetCapabilities(
|
|
XInputUserIndex dwUserIndex,
|
|
XInputCapabilitiesFlags dwFlags,
|
|
out XInputDeviceCapabilities pCapabilities);
|
|
|
|
[SuppressUnmanagedCodeSecurity]
|
|
internal delegate XInputErrorCode XInputGetState
|
|
(
|
|
XInputUserIndex dwUserIndex,
|
|
out XInputState pState
|
|
);
|
|
|
|
[SuppressUnmanagedCodeSecurity]
|
|
internal delegate XInputErrorCode XInputSetState
|
|
(
|
|
XInputUserIndex dwUserIndex,
|
|
ref XInputVibration pVibration
|
|
);
|
|
|
|
#endregion
|
|
|
|
#region IDisposable Members
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
void Dispose(bool manual)
|
|
{
|
|
if (manual)
|
|
{
|
|
if (dll != IntPtr.Zero)
|
|
{
|
|
Functions.FreeLibrary(dll);
|
|
dll = IntPtr.Zero;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IDisposable Members
|
|
|
|
public void Dispose()
|
|
{
|
|
Dispose(true);
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
void Dispose(bool manual)
|
|
{
|
|
if (manual)
|
|
{
|
|
xinput.Dispose();
|
|
}
|
|
else
|
|
{
|
|
Debug.Print("{0} leaked, did you forget to call Dispose()?", typeof(XInputJoystick).Name);
|
|
}
|
|
}
|
|
|
|
#if DEBUG
|
|
~XInputJoystick()
|
|
{
|
|
Dispose(false);
|
|
}
|
|
#endif
|
|
|
|
#endregion
|
|
}
|
|
}
|