diff --git a/Assembly-CSharp/Memoria/VoiceActing/BattleVoice.cs b/Assembly-CSharp/Memoria/VoiceActing/BattleVoice.cs index 383cdf2c0..90444390e 100644 --- a/Assembly-CSharp/Memoria/VoiceActing/BattleVoice.cs +++ b/Assembly-CSharp/Memoria/VoiceActing/BattleVoice.cs @@ -216,40 +216,42 @@ public static void TriggerOnBattleInOut(String when) if (isDirty) LoadEffects(); - try + List retainedEffects = new List(); + Int32 retainedPriority = Int32.MinValue; + foreach (BattleInOut effect in InOutEffect) { - List retainedEffects = new List(); - Int32 retainedPriority = Int32.MinValue; - foreach (BattleInOut effect in InOutEffect) + if (String.Compare(effect.When, when) != 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll()) + continue; + if (!String.IsNullOrEmpty(effect.Condition)) { - if (String.Compare(effect.When, when) != 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll()) - continue; - if (!String.IsNullOrEmpty(effect.Condition)) + Expression c = new Expression(effect.Condition); + BattleUnit unit = new BattleUnit(effect.Speakers[0].FindBtlUnlimited()); + NCalcUtility.InitializeExpressionUnit(ref c, unit); + c.Parameters["VictoryFocusIndex"] = (UInt32)VictoryFocusIndex; + c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; + c.EvaluateParameter += NCalcUtility.commonNCalcParameters; + + try { - Expression c = new Expression(effect.Condition); - BattleUnit unit = new BattleUnit(effect.Speakers[0].FindBtlUnlimited()); - NCalcUtility.InitializeExpressionUnit(ref c, unit); - c.Parameters["VictoryFocusIndex"] = (UInt32)VictoryFocusIndex; - c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; - c.EvaluateParameter += NCalcUtility.commonNCalcParameters; if (!NCalcUtility.EvaluateNCalcCondition(c.Evaluate())) continue; } - if (effect.Priority > retainedPriority) + catch (Exception err) { - retainedEffects.Clear(); - retainedPriority = effect.Priority; + Log.Error(err); + continue; } - retainedEffects.Add(effect); } - if (retainedEffects.Count == 0) - return; - PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); - } - catch (Exception err) - { - Log.Error(err); + if (effect.Priority > retainedPriority) + { + retainedEffects.Clear(); + retainedPriority = effect.Priority; + } + retainedEffects.Add(effect); } + if (retainedEffects.Count == 0) + return; + PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); } public static void TriggerOnBattleAct(BTL_DATA actingChar, String when, CMD_DATA cmdUsed, BattleCalculator calc = null) @@ -259,53 +261,54 @@ public static void TriggerOnBattleAct(BTL_DATA actingChar, String when, CMD_DATA if (isDirty) LoadEffects(); - try + List retainedEffects = new List(); + Int32 retainedPriority = Int32.MinValue; + foreach (BattleAct effect in ActEffect) { - List retainedEffects = new List(); - Int32 retainedPriority = Int32.MinValue; - foreach (BattleAct effect in ActEffect) + if (String.Compare(effect.When, when) != 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll() || !effect.CheckIsFirstSpeaker(actingChar)) + continue; + if (!String.IsNullOrEmpty(effect.Condition)) { - if (String.Compare(effect.When, when) != 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll() || !effect.CheckIsFirstSpeaker(actingChar)) - continue; - if (!String.IsNullOrEmpty(effect.Condition)) + Expression c = new Expression(effect.Condition); + BattleUnit unit = new BattleUnit(actingChar); + BattleCommand cmd = new BattleCommand(cmdUsed); + NCalcUtility.InitializeExpressionUnit(ref c, unit); + List t = FF9.btl_util.findAllBtlData(cmdUsed.tar_id); + if (t.Count > 0) + { + BattleUnit target = new BattleUnit(t[0]); + NCalcUtility.InitializeExpressionUnit(ref c, target, "Target"); + } + else + { + NCalcUtility.InitializeExpressionNullableUnit(ref c, null, "Target"); + } + NCalcUtility.InitializeExpressionCommand(ref c, cmd); + if (calc != null) // Should be the case only when "HitEffect" + NCalcUtility.InitializeExpressionAbilityContext(ref c, calc); + c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; + c.EvaluateParameter += NCalcUtility.commonNCalcParameters; + try { - Expression c = new Expression(effect.Condition); - BattleUnit unit = new BattleUnit(actingChar); - BattleCommand cmd = new BattleCommand(cmdUsed); - NCalcUtility.InitializeExpressionUnit(ref c, unit); - List t = FF9.btl_util.findAllBtlData(cmdUsed.tar_id); - if (t.Count > 0) - { - BattleUnit target = new BattleUnit(t[0]); - NCalcUtility.InitializeExpressionUnit(ref c, target, "Target"); - } - else - { - NCalcUtility.InitializeExpressionNullableUnit(ref c, null, "Target"); - } - NCalcUtility.InitializeExpressionCommand(ref c, cmd); - if (calc != null) // Should be the case only when "HitEffect" - NCalcUtility.InitializeExpressionAbilityContext(ref c, calc); - c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; - c.EvaluateParameter += NCalcUtility.commonNCalcParameters; if (!NCalcUtility.EvaluateNCalcCondition(c.Evaluate())) continue; } - if (effect.Priority > retainedPriority) + catch (Exception err) { - retainedEffects.Clear(); - retainedPriority = effect.Priority; + Log.Error(err); + continue; } - retainedEffects.Add(effect); } - if (retainedEffects.Count == 0) - return; - PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); - } - catch (Exception err) - { - Log.Error(err); + if (effect.Priority > retainedPriority) + { + retainedEffects.Clear(); + retainedPriority = effect.Priority; + } + retainedEffects.Add(effect); } + if (retainedEffects.Count == 0) + return; + PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); } public static void TriggerOnHitted(BTL_DATA hittedChar, BattleCalculator calc) @@ -315,42 +318,43 @@ public static void TriggerOnHitted(BTL_DATA hittedChar, BattleCalculator calc) if (isDirty) LoadEffects(); - try + List retainedEffects = new List(); + Int32 retainedPriority = Int32.MinValue; + foreach (BattleHitted effect in HittedEffect) { - List retainedEffects = new List(); - Int32 retainedPriority = Int32.MinValue; - foreach (BattleHitted effect in HittedEffect) + if (effect.Priority < retainedPriority || !effect.CheckSpeakerAll() || !effect.CheckIsFirstSpeaker(hittedChar)) + continue; + if (!String.IsNullOrEmpty(effect.Condition)) { - if (effect.Priority < retainedPriority || !effect.CheckSpeakerAll() || !effect.CheckIsFirstSpeaker(hittedChar)) - continue; - if (!String.IsNullOrEmpty(effect.Condition)) + Expression c = new Expression(effect.Condition); + BattleUnit unit = new BattleUnit(hittedChar); + NCalcUtility.InitializeExpressionUnit(ref c, unit); + NCalcUtility.InitializeExpressionUnit(ref c, calc.Caster, "Caster"); + NCalcUtility.InitializeExpressionCommand(ref c, calc.Command); + NCalcUtility.InitializeExpressionAbilityContext(ref c, calc); + c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; + c.EvaluateParameter += NCalcUtility.commonNCalcParameters; + try { - Expression c = new Expression(effect.Condition); - BattleUnit unit = new BattleUnit(hittedChar); - NCalcUtility.InitializeExpressionUnit(ref c, unit); - NCalcUtility.InitializeExpressionUnit(ref c, calc.Caster, "Caster"); - NCalcUtility.InitializeExpressionCommand(ref c, calc.Command); - NCalcUtility.InitializeExpressionAbilityContext(ref c, calc); - c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; - c.EvaluateParameter += NCalcUtility.commonNCalcParameters; if (!NCalcUtility.EvaluateNCalcCondition(c.Evaluate())) continue; } - if (effect.Priority > retainedPriority) + catch (Exception err) { - retainedEffects.Clear(); - retainedPriority = effect.Priority; + Log.Error(err); + continue; } - retainedEffects.Add(effect); } - if (retainedEffects.Count == 0) - return; - PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); - } - catch (Exception err) - { - Log.Error(err); + if (effect.Priority > retainedPriority) + { + retainedEffects.Clear(); + retainedPriority = effect.Priority; + } + retainedEffects.Add(effect); } + if (retainedEffects.Count == 0) + return; + PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); } public static void TriggerOnStatusChange(BTL_DATA statusedChar, String when, BattleStatus whichStatus) @@ -360,44 +364,45 @@ public static void TriggerOnStatusChange(BTL_DATA statusedChar, String when, Bat if (isDirty) LoadEffects(); - try + List retainedEffects = new List(); + Int32 retainedPriority = Int32.MinValue; + Boolean discardStatusChecks = String.Compare(when, "Removed") != 0; + foreach (BattleStatusChange effect in StatusChangeEffect) { - List retainedEffects = new List(); - Int32 retainedPriority = Int32.MinValue; - Boolean discardStatusChecks = String.Compare(when, "Removed") != 0; - foreach (BattleStatusChange effect in StatusChangeEffect) + if (String.Compare(effect.When, when) != 0 || (whichStatus & effect.Status) == 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll(statusedChar, effect.Status)) + continue; + if (discardStatusChecks && !effect.CheckIsFirstSpeaker(statusedChar, effect.Status)) + continue; + if (!discardStatusChecks && !effect.CheckIsFirstSpeaker(statusedChar)) + continue; + if (!String.IsNullOrEmpty(effect.Condition)) { - if (String.Compare(effect.When, when) != 0 || (whichStatus & effect.Status) == 0 || effect.Priority < retainedPriority || !effect.CheckSpeakerAll(statusedChar, effect.Status)) - continue; - if (discardStatusChecks && !effect.CheckIsFirstSpeaker(statusedChar, effect.Status)) - continue; - if (!discardStatusChecks && !effect.CheckIsFirstSpeaker(statusedChar)) - continue; - if (!String.IsNullOrEmpty(effect.Condition)) + Expression c = new Expression(effect.Condition); + BattleUnit unit = new BattleUnit(statusedChar); + NCalcUtility.InitializeExpressionUnit(ref c, unit); + c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; + c.EvaluateParameter += NCalcUtility.commonNCalcParameters; + try { - Expression c = new Expression(effect.Condition); - BattleUnit unit = new BattleUnit(statusedChar); - NCalcUtility.InitializeExpressionUnit(ref c, unit); - c.EvaluateFunction += NCalcUtility.commonNCalcFunctions; - c.EvaluateParameter += NCalcUtility.commonNCalcParameters; if (!NCalcUtility.EvaluateNCalcCondition(c.Evaluate())) continue; } - if (effect.Priority > retainedPriority) + catch (Exception err) { - retainedEffects.Clear(); - retainedPriority = effect.Priority; + Log.Error(err); + continue; } - retainedEffects.Add(effect); } - if (retainedEffects.Count == 0) - return; - PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); - } - catch (Exception err) - { - Log.Error(err); + if (effect.Priority > retainedPriority) + { + retainedEffects.Clear(); + retainedPriority = effect.Priority; + } + retainedEffects.Add(effect); } + if (retainedEffects.Count == 0) + return; + PlayVoiceEffect(retainedEffects[UnityEngine.Random.Range(0, retainedEffects.Count)]); } private static void ParseEffect(String effectCode) diff --git a/Assembly-CSharp/NCalc/NCalcUtility.cs b/Assembly-CSharp/NCalc/NCalcUtility.cs index 5104fe0af..fba5c3fa7 100644 --- a/Assembly-CSharp/NCalc/NCalcUtility.cs +++ b/Assembly-CSharp/NCalc/NCalcUtility.cs @@ -96,7 +96,7 @@ public static String EvaluateNCalcString(Object obj, String defaultResult = "") else if (name == "IsCharacterInParty" && args.Parameters.Length == 1) { CharacterId index = (CharacterId)NCalcUtility.ConvertNCalcResult(args.Parameters[0].Evaluate(), Byte.MaxValue); - args.Result = index != CharacterId.NONE && FF9StateSystem.Common.FF9.party.member.Any(p => p.Index == index); + args.Result = index != CharacterId.NONE && FF9StateSystem.Common.FF9.party.member.Any(p => p?.Index == index); } else if (name == "GetCategoryKillCount" && args.Parameters.Length == 1) { diff --git a/Assembly-CSharp/UnityXInput/Input.cs b/Assembly-CSharp/UnityXInput/Input.cs index 658116e3e..e44562897 100644 --- a/Assembly-CSharp/UnityXInput/Input.cs +++ b/Assembly-CSharp/UnityXInput/Input.cs @@ -275,41 +275,28 @@ private static Boolean GetXButtonDown(String keyName) public static Single GetAxis(String axisName) { - Single axis = UnityEngine.Input.GetAxis(axisName); - if ((Double)axis != 0.0) - return axis; - return Input.GetXAxis(axisName); + return Input.GetAxisRaw(axisName); } public static Single GetAxisRaw(String axisName) { - Single axisRaw = UnityEngine.Input.GetAxisRaw(axisName); - if ((Double)axisRaw != 0.0) - return axisRaw; - return Input.GetXAxis(axisName); + float axisRaw = UnityEngine.Input.GetAxisRaw(axisName); + axisRaw += Input.GetXAxis(axisName); + return Mathf.Clamp(axisRaw, -1f, 1f); } public static Boolean GetButtonUp(String keyName) { - Boolean buttonUp = UnityEngine.Input.GetButtonUp(keyName); - if (buttonUp) - return buttonUp; return Input.GetXButtonUp(keyName); } public static Boolean GetButtonDown(String keyName) { - Boolean buttonDown = UnityEngine.Input.GetButtonDown(keyName); - if (buttonDown) - return buttonDown; return Input.GetXButtonDown(keyName); } public static Boolean GetButton(String keyName) { - Boolean button = UnityEngine.Input.GetButton(keyName); - if (button) - return button; return Input.GetXButton(keyName); } diff --git a/Memoria.MSBuild/Pack.cs b/Memoria.MSBuild/Pack.cs index 959bcdbe3..b69c0e2ed 100644 --- a/Memoria.MSBuild/Pack.cs +++ b/Memoria.MSBuild/Pack.cs @@ -62,7 +62,9 @@ public Boolean Execute() PackOptionalFile("Launcher\\Memoria.Launcher.exe.config", "FF9_Launcher.exe.config", compressStream, bw, pathMap, ref uncompressedDataSize); PackOptionalFile("Launcher\\Memoria.SteamFix.exe", "Memoria.SteamFix.exe", compressStream, bw, pathMap, ref uncompressedDataSize); PackOptionalFile("Launcher\\Memoria.ini", "Memoria.ini", compressStream, bw, pathMap, ref uncompressedDataSize); - + PackOptionalFile("XInputDotNetPure.dll", "{PLATFORM}\\FF9_Data\\Managed\\XInputDotNetPure.dll", compressStream, bw, pathMap, ref uncompressedDataSize); + PackOptionalFile("JoyShockLibrary\\x64\\JoyShockLibrary.dll", "x64\\FF9_Data\\Plugins\\JoyShockLibrary.dll", compressStream, bw, pathMap, ref uncompressedDataSize); + PackOptionalFile("JoyShockLibrary\\x86\\JoyShockLibrary.dll", "x86\\FF9_Data\\Plugins\\JoyShockLibrary.dll", compressStream, bw, pathMap, ref uncompressedDataSize); bw.Flush(); Int64 compressedDataSize = executableFile.Position - compressedDataPosition; diff --git a/Memoria.XInputDotNetPure/GamePad.cs b/Memoria.XInputDotNetPure/GamePad.cs new file mode 100644 index 000000000..2b1605574 --- /dev/null +++ b/Memoria.XInputDotNetPure/GamePad.cs @@ -0,0 +1,439 @@ +using System; +using System.IO; +using System.Runtime.InteropServices; +using static XInputDotNetPure.GamePadState; + +namespace XInputDotNetPure +{ + class Imports + { + internal const string DLLName = "XInputInterface"; + + [DllImport(DLLName)] + public static extern uint XInputGamePadGetState(uint playerIndex, out GamePadState.RawState state); + [DllImport(DLLName)] + public static extern void XInputGamePadSetState(uint playerIndex, float leftMotor, float rightMotor); + } + + public enum ButtonState + { + Pressed, + Released + } + + public struct GamePadButtons + { + ButtonState start, back, leftStick, rightStick, leftShoulder, rightShoulder, guide, a, b, x, y; + + internal GamePadButtons(ButtonState start, ButtonState back, ButtonState leftStick, ButtonState rightStick, + ButtonState leftShoulder, ButtonState rightShoulder, ButtonState guide, + ButtonState a, ButtonState b, ButtonState x, ButtonState y) + { + this.start = start; + this.back = back; + this.leftStick = leftStick; + this.rightStick = rightStick; + this.leftShoulder = leftShoulder; + this.rightShoulder = rightShoulder; + this.guide = guide; + this.a = a; + this.b = b; + this.x = x; + this.y = y; + } + + public ButtonState Start + { + get { return start; } + } + + public ButtonState Back + { + get { return back; } + } + + public ButtonState LeftStick + { + get { return leftStick; } + } + + public ButtonState RightStick + { + get { return rightStick; } + } + + public ButtonState LeftShoulder + { + get { return leftShoulder; } + } + + public ButtonState RightShoulder + { + get { return rightShoulder; } + } + + public ButtonState Guide + { + get { return guide; } + } + + public ButtonState A + { + get { return a; } + } + + public ButtonState B + { + get { return b; } + } + + public ButtonState X + { + get { return x; } + } + + public ButtonState Y + { + get { return y; } + } + } + + public struct GamePadDPad + { + ButtonState up, down, left, right; + + internal GamePadDPad(ButtonState up, ButtonState down, ButtonState left, ButtonState right) + { + this.up = up; + this.down = down; + this.left = left; + this.right = right; + } + + public ButtonState Up + { + get { return up; } + } + + public ButtonState Down + { + get { return down; } + } + + public ButtonState Left + { + get { return left; } + } + + public ButtonState Right + { + get { return right; } + } + } + + public struct GamePadThumbSticks + { + public struct StickValue + { + float x, y; + + internal StickValue(float x, float y) + { + this.x = x; + this.y = y; + } + + public float X + { + get { return x; } + } + + public float Y + { + get { return y; } + } + } + + StickValue left, right; + + internal GamePadThumbSticks(StickValue left, StickValue right) + { + this.left = left; + this.right = right; + } + + public StickValue Left + { + get { return left; } + } + + public StickValue Right + { + get { return right; } + } + } + + public struct GamePadTriggers + { + float left; + float right; + + internal GamePadTriggers(float left, float right) + { + this.left = left; + this.right = right; + } + + public float Left + { + get { return left; } + } + + public float Right + { + get { return right; } + } + } + + public struct GamePadState + { + [StructLayout(LayoutKind.Sequential)] + internal struct RawState + { + public uint dwPacketNumber; + public GamePad Gamepad; + + [StructLayout(LayoutKind.Sequential)] + public struct GamePad + { + public ushort wButtons; + public byte bLeftTrigger; + public byte bRightTrigger; + public short sThumbLX; + public short sThumbLY; + public short sThumbRX; + public short sThumbRY; + } + } + + bool isConnected; + uint packetNumber; + GamePadButtons buttons; + GamePadDPad dPad; + GamePadThumbSticks thumbSticks; + GamePadTriggers triggers; + + public enum ButtonsConstants + { + DPadUp = 0x00000001, + DPadDown = 0x00000002, + DPadLeft = 0x00000004, + DPadRight = 0x00000008, + Start = 0x00000010, + Back = 0x00000020, + LeftThumb = 0x00000040, + RightThumb = 0x00000080, + LeftShoulder = 0x0100, + RightShoulder = 0x0200, + Guide = 0x0400, + A = 0x1000, + B = 0x2000, + X = 0x4000, + Y = 0x8000 + } + + internal GamePadState(bool isConnected, RawState rawState, GamePadDeadZone deadZone) + { + this.isConnected = isConnected; + + if (!isConnected) + { + rawState.dwPacketNumber = 0; + rawState.Gamepad.wButtons = 0; + rawState.Gamepad.bLeftTrigger = 0; + rawState.Gamepad.bRightTrigger = 0; + rawState.Gamepad.sThumbLX = 0; + rawState.Gamepad.sThumbLY = 0; + rawState.Gamepad.sThumbRX = 0; + rawState.Gamepad.sThumbRY = 0; + } + + packetNumber = rawState.dwPacketNumber; + buttons = new GamePadButtons( + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.Start) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.Back) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.LeftThumb) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.RightThumb) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.LeftShoulder) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.RightShoulder) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.Guide) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.A) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.B) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.X) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.Y) != 0 ? ButtonState.Pressed : ButtonState.Released + ); + dPad = new GamePadDPad( + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.DPadUp) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.DPadDown) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.DPadLeft) != 0 ? ButtonState.Pressed : ButtonState.Released, + (rawState.Gamepad.wButtons & (uint)ButtonsConstants.DPadRight) != 0 ? ButtonState.Pressed : ButtonState.Released + ); + + thumbSticks = new GamePadThumbSticks( + Utils.ApplyLeftStickDeadZone(rawState.Gamepad.sThumbLX, rawState.Gamepad.sThumbLY, deadZone), + Utils.ApplyRightStickDeadZone(rawState.Gamepad.sThumbRX, rawState.Gamepad.sThumbRY, deadZone) + ); + triggers = new GamePadTriggers( + Utils.ApplyTriggerDeadZone(rawState.Gamepad.bLeftTrigger, deadZone), + Utils.ApplyTriggerDeadZone(rawState.Gamepad.bRightTrigger, deadZone) + ); + } + + public uint PacketNumber + { + get { return packetNumber; } + } + + public bool IsConnected + { + get { return isConnected; } + } + + public GamePadButtons Buttons + { + get { return buttons; } + } + + public GamePadDPad DPad + { + get { return dPad; } + } + + public GamePadTriggers Triggers + { + get { return triggers; } + } + + public GamePadThumbSticks ThumbSticks + { + get { return thumbSticks; } + } + } + + public enum PlayerIndex + { + One = 0, + Two, + Three, + Four + } + + public enum GamePadDeadZone + { + Circular, + IndependentAxes, + None + } + + public class GamePad + { + public static GamePadState GetState(PlayerIndex playerIndex) + { + return GetState(playerIndex, GamePadDeadZone.IndependentAxes); + } + + public static GamePadState GetState(PlayerIndex playerIndex, GamePadDeadZone deadZone) + { + RawState rawState; + uint result = Imports.XInputGamePadGetState((uint)playerIndex, out rawState); + + try + { + if (jslDevices == null) + { + // ConnectDevices can hang the game in some cases, so it's only called at startup + // Detecting system device changes would be better + int c = JSL.ConnectDevices(); + jslDevices = new int[c > 0 ? c : 0]; + JSL.GetConnectedDeviceHandles(jslDevices, jslDevices.Length); + } + if (jslDevices.Length > 0) + { + foreach (int device in jslDevices) + { + if (!JSL.StillConnected(device)) continue; + + // We combine the inputs from all the connected gamepads + // Applying deazone before doing so would be better but who has more than one gamepad connected anyway + JSL.State jslState = JSL.GetSimpleState(device); + if ((jslState.buttons & (int)JSL.Button.Options) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.Start; + if ((jslState.buttons & (int)JSL.Button.TouchpadClick) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.Back; + if ((jslState.buttons & (int)JSL.Button.Minus) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.Back; + if ((jslState.buttons & (int)JSL.Button.ZL) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.LeftThumb; + if ((jslState.buttons & (int)JSL.Button.ZR) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.RightThumb; + if ((jslState.buttons & (int)JSL.Button.L) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.LeftShoulder; + if ((jslState.buttons & (int)JSL.Button.R) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.RightShoulder; + if ((jslState.buttons & (int)JSL.Button.PS) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.Guide; + if ((jslState.buttons & (int)JSL.Button.S) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.A; + if ((jslState.buttons & (int)JSL.Button.E) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.B; + if ((jslState.buttons & (int)JSL.Button.W) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.X; + if ((jslState.buttons & (int)JSL.Button.N) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.Y; + + if ((jslState.buttons & (int)JSL.Button.Up) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.DPadUp; + if ((jslState.buttons & (int)JSL.Button.Down) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.DPadDown; + if ((jslState.buttons & (int)JSL.Button.Left) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.DPadLeft; + if ((jslState.buttons & (int)JSL.Button.Right) != 0) rawState.Gamepad.wButtons |= (ushort)ButtonsConstants.DPadRight; + + rawState.Gamepad.sThumbLX = (short)Utils.Clamp((int)(jslState.stickLX * short.MaxValue + rawState.Gamepad.sThumbLX), short.MinValue, short.MaxValue); + rawState.Gamepad.sThumbLY = (short)Utils.Clamp((int)(jslState.stickLY * short.MaxValue + rawState.Gamepad.sThumbLY), short.MinValue, short.MaxValue); + + rawState.Gamepad.sThumbRX = (short)Utils.Clamp((int)(jslState.stickRX * short.MaxValue + rawState.Gamepad.sThumbRX), short.MinValue, short.MaxValue); + rawState.Gamepad.sThumbRY = (short)Utils.Clamp((int)(jslState.stickRY * short.MaxValue + rawState.Gamepad.sThumbRY), short.MinValue, short.MaxValue); + + rawState.Gamepad.bLeftTrigger = (byte)Utils.Clamp((int)(jslState.lTrigger * byte.MaxValue + rawState.Gamepad.bLeftTrigger), byte.MinValue, byte.MaxValue); + rawState.Gamepad.bRightTrigger = (byte)Utils.Clamp((int)(jslState.rTrigger * byte.MaxValue + rawState.Gamepad.bRightTrigger), byte.MinValue, byte.MaxValue); + } + result = Utils.Success; + } + } + catch (Exception e) + { + Log(e.ToString()); + } + return new GamePadState(result == Utils.Success, rawState, deadZone); + } + + public static void SetVibration(PlayerIndex playerIndex, float leftMotor, float rightMotor) + { + Imports.XInputGamePadSetState((uint)playerIndex, leftMotor, rightMotor); + try + { + if (jslDevices.Length > 0) + { + foreach (int device in jslDevices) + { + if (!JSL.StillConnected(device)) continue; + JSL.SetRumble(device, (int)(leftMotor * byte.MaxValue), (int)(rightMotor * byte.MaxValue)); + } + } + } + catch (Exception e) + { + Log(e.ToString()); + } + } + + private static int[] jslDevices = null; + private static long filePosition = 0; + private static void Log(string message) + { + using (var file = File.OpenWrite("XInpuDotNetPure.log")) + using (var log = new StreamWriter(file)) + { + file.Seek(filePosition, SeekOrigin.Begin); + log.WriteLine(message); + filePosition = file.Position; + } + } + } +} diff --git a/Memoria.XInputDotNetPure/JoyShockLibrary.cs b/Memoria.XInputDotNetPure/JoyShockLibrary.cs new file mode 100644 index 000000000..d98d88d02 --- /dev/null +++ b/Memoria.XInputDotNetPure/JoyShockLibrary.cs @@ -0,0 +1,57 @@ +using System.Runtime.InteropServices; + +public static class JSL +{ + [StructLayout(LayoutKind.Sequential)] + public struct State + { + public int buttons; + public float lTrigger; + public float rTrigger; + public float stickLX; + public float stickLY; + public float stickRX; + public float stickRY; + } + + public enum Button + { + Up = 1, + Down = 1 << 1, + Left = 1 << 2, + Right = 1 << 3, + Plus = 1 << 4, + Options = 1 << 4, + Minus = 1 << 5, + Share = 1 << 5, + LClick = 1 << 6, + RClick = 1 << 7, + L = 1 << 8, + R = 1 << 9, + ZL = 1 << 10, + ZR = 1 << 11, + S = 1 << 12, + E = 1 << 13, + W = 1 << 14, + N = 1 << 15, + Home = 1 << 16, + PS = 1 << 16, + Capture = 1 << 17, + TouchpadClick = 1 << 17, + SL = 1 << 18, + SR = 1 << 19 + } + + private const string Library = "JoyShockLibrary"; + + [DllImport(Library, EntryPoint = "JslConnectDevices")] + public static extern int ConnectDevices(); + [DllImport(Library, EntryPoint = "JslStillConnected")] + public static extern bool StillConnected(int deviceId); + [DllImport(Library, EntryPoint = "JslGetConnectedDeviceHandles")] + public static extern int GetConnectedDeviceHandles(int[] deviceHandleArray, int size); + [DllImport(Library, EntryPoint = "JslGetSimpleState", CallingConvention = CallingConvention.Cdecl)] + public static extern State GetSimpleState(int deviceId); + [DllImport(Library, EntryPoint = "JslSetRumble")] + public static extern void SetRumble(int deviceId, int smallRumble, int bigRumble); +} \ No newline at end of file diff --git a/Memoria.XInputDotNetPure/JoyShockLibrary/LICENSE.md b/Memoria.XInputDotNetPure/JoyShockLibrary/LICENSE.md new file mode 100644 index 000000000..f9b0e86e7 --- /dev/null +++ b/Memoria.XInputDotNetPure/JoyShockLibrary/LICENSE.md @@ -0,0 +1,31 @@ +### MIT License + +Copyright 2018-2023 Julian Smart + +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. + +--- + +This software incorporates some work from the following projects and their copyright holders, all also covered by the same MIT license, with the same permissions and disclaimers: +* mfosse's JoyCon-Driver (JoyCon driver), Copyright 2018 Matthew Fosse: https://github.com/mfosse/JoyCon-Driver +* chrippa's ds4drv (DualShock 4 driver), Copyright 2013-2014 Christopher Rosell: https://github.com/chrippa/ds4drv +* Ryochan7 and Jay2Kings' DS4Windows (DualShock 4 input mapper), Copyright 2019 Travis Nickles: https://github.com/Ryochan7/DS4Windows +* My GamepadMotionHelpers (gravity calculation and gyro calibration), Copyright 2020-2021 Julian Smart: https://github.com/JibbSmart/GamepadMotionHelpers + +--- + +This software also links to signal11's HIDAPI. HIDAPI has its own permissive license: https://github.com/signal11/hidapi + +HIDAPI - Multi-Platform library for +communication with HID devices. + +Copyright 2009, Alan Ott, Signal 11 Software. +All Rights Reserved. + +This software may be used by anyone for any reason so +long as the copyright notice in the source files +remains intact. \ No newline at end of file diff --git a/Memoria.XInputDotNetPure/JoyShockLibrary/README.md b/Memoria.XInputDotNetPure/JoyShockLibrary/README.md new file mode 100644 index 000000000..e9b2aeedf --- /dev/null +++ b/Memoria.XInputDotNetPure/JoyShockLibrary/README.md @@ -0,0 +1,216 @@ +# JoyShockLibrary +The Sony PlayStation's DualShock 4, DualSense, Nintendo Switch Joy-Cons (used in pairs), and Nintendo Switch Pro Controller have much in common. They have many of the features expected of modern game controllers. They also have an incredibly versatile and underutilised input that their biggest rival (Microsoft's Xbox One controller) doesn't have: the gyro. + +My goal with JoyShockLibrary is to enable game developers to support DS4, DS, Joy-Cons, and Pro Controllers natively in PC games. I've compiled the library for Windows, but it uses platform-agnostic tools, and my hope is that other developers would be able to get it working on other platforms (such as Linux or Mac) without too much trouble. + +## Contents +* **[Releases](#releases)** +* **[Reference](#reference)** + * **[Structs](#structs)** + * **[Functions](#functions)** +* **[Known and Perceived Issues](#known-and-perceived-issues)** +* **[Backwards Compatibility](#backwards-compatibility)** +* **[Credits](#credits)** +* **[Helpful Resources](#helpful-resources)** +* **[License](#license)** + +## Releases +The latest version of JoyShockLibrary can always be found [here](https://github.com/JibbSmart/JoyShockLibrary/releases). Included is a 64-bit dll and a 32-bit dll, both for Windows, and JoyShockLibrary.h and JoyShockLibrary.cs for using the dll in C/C++ and C\# (Unity), respectively. The .cs file isn't up to date with the latest features, but should provide a starting point for your use. + +## Reference +*JoyShockLibrary.h* has everything you need to use the library, but here's a breakdown of everything in there. + +### Structs +**struct JOY\_SHOCK\_STATE** - This struct contains the state for all the sticks, buttons, and triggers on the controller. If you're just using JoyShockLibrary to be able to use Joy-Cons, Pro Controllers, DualSenses, and DualShock 4s similarly to how you'd use other devices, this has everything you need to know. +* **int buttons** contains the states of all the controller's buttons with the following masks: + * ```0x00001``` - d-pad ```up``` + * ```0x00002``` - d-pad ```down``` + * ```0x00004``` - d-pad ```left``` + * ```0x00008``` - d-pad ```right``` + * ```0x00010``` - ```+``` on Nintendo devices, ```Options``` on DS4 + * ```0x00020``` - ```-``` on Nintendo devices, ```Share``` on DS4 + * ```0x00040``` - ```left-stick click``` on Nintendo devices, ```L3``` on DS4 + * ```0x00080``` - ```right-stick click``` on Nintendo devices, ```R3``` on DS4 + * ```0x00100``` - ```L``` on Nintendo devices, ```L1``` on DS4 + * ```0x00200``` - ```R``` on Nintendo devices, ```R1``` on DS4 + * ```0x00400``` - ```ZL``` on Nintendo devices, ```L2``` on DS4 + * ```0x00800``` - ```ZR``` on Nintendo devices, ```R2``` on DS4 + * ```0x01000``` - the South face-button: ```B``` on Nintendo devices, ```⨉``` on DS4 + * ```0x02000``` - the East face-button: ```A``` on Nintendo devices, ```○``` on DS4 + * ```0x04000``` - the West face-button: ```Y``` on Nintendo devices, ```□``` on DS4 + * ```0x08000``` - the North face-button: ```X``` on Nintendo devices, ```△``` on DS4 + * ```0x10000``` - ```Home``` on Nintendo devices, ```PS``` on DS4 + * ```0x20000``` - ```Capture``` on Nintendo devices, ```touchpad click``` on DS4, ```Create``` on DS5 + * ```0x40000``` - ```SL``` on Nintendo Joy-Cons, ```Mic``` on DS5 + * ```0x80000``` - ```SR``` on Nintendo Joy-Cons +* **float lTrigger** - how far has the left trigger been pressed? This will be 1 or 0 on Nintendo devices, which don't have analog triggers +* **float rTrigger** - how far has the right trigger been pressed? This will be 1 or 0 on Nintendo devices, which don't have analog triggers +* **float stickLX, stickLY** - left-stick X axis and Y axis, respectively, from -1 to 1 +* **float stickRX, stickRX** - right-stick X axis and Y axis, respectively, from -1 to 1 + +**struct IMU_STATE** - Each supported device contains an IMU which has a 3-axis accelerometer and a 3-axis gyroscope. IMU\_STATE is where you find that info. +* **float accelX, accelY, accelZ** - accelerometer X axis, Y axis, and Z axis, respectively, in g (g-force). +* **float gyroX, gyroY, gyroZ** - gyroscope angular velocity X, Y, and Z, respectively, in dps (degrees per second), when correctly calibrated. + +**struct MOTION_STATE** - The MOTION_STATE reports the orientation of the device as calculated using a sensor fusion solution to combine gyro and accelerometer data. +* **float quatW, quatX, quatY, quatZ** - a quaternion representing the orientation of the device. +* **float accelX, accelY, accelZ** - local acceleration after accounting for and removing the effect of gravity. +* **float gravX, gravY, gravZ** - local gravity direction. + +#### Tip +Quaternions are useful if you want to try and represent the device's 3D orientation in-game, with one major limitation: "yaw drift". Small errors will accumulate over time so that the quaternion orientation no longer matches the controller's real orientation. The gravity direction is used to counter this error in the roll and pitch axes, but there's nothing for countering the error in the yaw axis. + +Quaternions are **not** recommended for mouse-like aiming or cursor control. The gravity correction applied to the quaternion is not useful for these cases, and only introduces more error. For these, it's much better to use the calibrated gyro angular velocities. To make sure you account for every motion sensor update, either set a callback where you can respond to new motion input as it comes in, or poll the controller using **JslGetAndFlushAccumulatedGyro** to get a good average angular velocity since you last called this function. + +### Functions + +All these functions *should* be thread-safe, and none of them should cause any harm if given the wrong or out-of-date deviceId. So even if a device gets disconnected between calling "JslStillConnected" and "JslGetSimpleState", the latter will just report all the sticks, triggers, and buttons untouched, and you'll detect the disconnection *next time* you call "JslStillConnected". + +**int JslConnectDevices()** - Register any connected devices. Returns the number of devices connected, which is helpful for getting the handles for those devices with the next function. As of version 3, this will not interrupt current connections. So you can call this function at any time to check for new connections. To only call it when necessary, only do this when your OS notifies you of a new connection (eg WM_DEVICECHANGE on Windows). + +**int JslGetConnectedDeviceHandles(int\* deviceHandleArray, int size)** - Fills the array *deviceHandleArray* of size *size* with the handles for all connected devices, up to the length of the array. Use the length returned by *JslConnectDevices* to make sure you've got all connected devices' handles. + +**void JslDisconnectAndDisposeAll()** - Disconnect devices, no longer polling them for input. + +**bool JslStillConnected(int deviceId)** - Returns **true** if the controller with the given id is still connected. + +**JOY\_SHOCK\_STATE JslGetSimpleState(int deviceId)** - Get the latest button + trigger + stick state for the controller with the given id. + +**IMU\_STATE JslGetIMUState(int deviceId)** - Get the latest accelerometer + gyroscope state for the controller with the given id. + +**MOTION\_STATE JslGetMotionState(int deviceId)** - Get the latest motion state for the controller with the given id. + +**TOUCH\_STATE JslGetTouchState(int deviceId, bool previous = false)** - Get the latest or previous touchpad state for the controller with the given id. Only DualShock 4 and DualSense support this. + +**bool JslGetTouchpadDimension(int deviceId, int &sizeX, int &sizeY)** - Get the dimension of the touchpad. This is useful to abstract the resolution of different touchpads. + +**int JslGetButtons(int deviceId)** - Get the latest button state for the controller with the given id. If you want more than just the buttons, it's more efficient to use JslGetSimpleState. + +**float JslGetLeftX/JslGetLeftY/JslGetRightX/JslGetRightY(int deviceId)** - Get the latest stick state for the controller with the given id. If you want more than just a single stick axis, it's more efficient to use JslGetSimpleState. + +**float JslGetLeftTrigger/JslGetRightTrigger(int deviceId)** - Get the latest trigger state for the controller with the given id. If you want more than just a single trigger, it's more efficient to use JslGetSimpleState. + +**float JslGetGyroX/JslGetGyroY/JslGetGyroZ(int deviceId)** - Get the latest angular velocity for a given gyroscope axis. If you want more than just a single gyroscope axis velocity, it's more efficient to use JslGetIMUState. + +**void JslGetAndFlushAccumulatedGyro(int deviceId, float& gyroX, float& gyroY, float& gyroZ)** - Get gyro input that has accumulated since you last called this function on the device with the given id. This is an average angular velocity. Highly recommended if you're polling controllers instead of using the callbacks (_JslSetCallback_) to get new gyro data the moment it is received, to give a better account of how the controller has been turned since you last polled it. If no new motion data has come in since you last called it, it will repeat the same values, to avoid stuttery motion input when your game's framerate is higher than the controller's reporting rate (common with Switch controllers on high end PCs, for example). + +**void JslSetGyroSpace(int deviceId, int gyroSpace)** - Players have different ideas about what the front of the controller is, or what the controller's rest position should be. This can make it hard to decide whether to use the controller's yaw or roll axis for turning in-game or moving a cursor horizontally. To address this, there are 3 popular "spaces" for gyro controls: +- 0 = Local Space - just pick yaw or roll and potentially give the player the option to change it. Or add them together. Up to you. It's not doing anything, so this is the default. +- 1 = World Space - use the gravity calculation to figure out which way is "up", and use that to calculate new axes of rotation for horizontal and vertical movement. Adapts to player preferences, but acquires some error from depending so strongly on the gravity calculation. +- 2 = Player Space - pretty much as adaptive as World Space, but uses gravity more loosely, so it doesn't acquire any error from errors in the gravity calculation. A great default for standard controllers (where the screen isn't attached to the controller). But it's not the default here. + +JslSetGyroSpace lets you choose one of those spaces, and the transformation will automatically be applied when you request gyro (JslGetGyro* functions), get the IMU_STATE, or receive the IMU_STATE in a callback. + +**float JslGetAccelX/JslGetAccelY/JslGetAccelZ(int deviceId)** - Get the latest acceleration for a given axis. If you want more than just a accelerometer axis, it's more efficient to use JslGetIMUState. + +**int JslGetTouchId(int deviceId, bool secondTouch=false)** - Get the last touch's id, which is a value in range of 0-127 that automaticaly increments whenever a new touch appears, for the controller with the given id. Only DualShock 4s support this. If you want more than just a touch's id, it's more efficient to use JslGetTouchState. + +**bool JslGetTouchDown(int deviceId, bool secondTouch=false)** - Get the latest state of the touch being present on a touchpad for the controller with the given id. Only DualShock 4s support this. If you want more than just a presence of touch, it's more efficient to use JslGetTouchState. + +**float JslGetTouchX/JslGetTouchY(int deviceId, bool secondTouch=false)** - Get the latest touch state for the controller with the given id. Only DualShock 4s support this. If you want more than just a single touch axis, it's more efficient to use JslGetTouchState. + +**float JslGetStickStep(int deviceId)** - Different devices use different size data types and different ranges on those data types when reporting stick axes. For some calculations, it may be important to know the limits of the current device and work around them in different ways. This gives the smallest step size between two values for the given device's analog sticks. + +**float JslGetTriggerStep(int deviceId)** - Some devices have analog triggers, some don't. For some calculations, it may be important to know the limits of the current device and work around them in different ways. This gives the smallest step size between two values for the given device's triggers, or 1.0 if they're actually just binary inputs. + +**float JslGetTriggerStep(int deviceId)** - Some devices have analog triggers, some don't. For some calculations, it may be important to know the limits of the current device and work around them in different ways. This gives the smallest step size between two values for the given device's triggers, or 1.0 if they're actually just binary inputs. + +**float JslGetPollRate(int deviceId)** - Different devices report back new information at different rates. For the given device, this gives how many times one would usually expect the device to report back per second. + +**float JslGetTimeSinceLastUpdate(int deviceId)** - Getting the time since the last update was received (in seconds) can be helpful for communicating a poor connection to the user (which can help communicate to wireless players that they need to move closer to the console or in some other way improve the connection, as [Fly Together](https://youtu.be/BjksCGFknKo?t=782) communicates so well in this example). + +**void JslResetContinuousCalibration(int deviceId)** - JoyShockLibrary has helpful functions for calibrating the gyroscope by averaging out its input over time. This deletes all calibration data that's been accumulated, if any, this session. + +**void JslStartContinuousCalibration(int deviceId)** - Start collecting gyro data, recording the ongoing average and using that to offset gyro output. + +**void JslPauseContinuousCalibration(int deviceId)** - Stop collecting gyro data, but don't delete it. + +**void JslSetAutomaticCalibration(int deviceId, bool enabled)** - Enable to have gyro automatically recalibrated when the controller is at rest or held very still. Disable to return to manual calibration using the above functions. + +**void JslGetCalibrationOffset(int deviceId, float& xOffset, float& yOffset, float& zOffset)** - Get the calibrated offset value for the given device's gyro. You don't have to use it; all gyro output for this device is already being offset by this vector before leaving JoyShockLibrary. + +**void JslSetCalibrationOffset(int deviceId, float xOffset, float yOffset, float zOffset)** - Manually set the calibrated offset value for the given device's gyro. + +**JSL_AUTO_CALIBRATION JslGetAutoCalibrationStatus(int deviceId)** - Get whether auto calibration is enabled, whether we think the controller is currently being held still, and how confident we are in its auto calibration. If you want to prompt the user to manually calibrate their controller, you can use auto calibration, and read these values to show to the user whether the controller is at rest as well as progress for calibration. Then, you can either leave auto calibration enabled (when confidence is high, auto calibration becomes difficult to trigger accidentally), or just disable auto calibration to prevent further changes without the user triggering it manually again. + +**void JslSetCallback(void(\*callback)(int, JOY\_SHOCK\_STATE, JOY\_SHOCK\_STATE, IMU\_STATE, IMU\_STATE, float))** - Set a callback function by which JoyShockLibrary can report the current state for each device. This callback will be given the *deviceId* for the reporting device, its current button + trigger + stick state, its previous button + trigger + stick state, its current accelerometer + gyro state, its previous accelerometer + gyro state, and the amount of time since the last report for this device (in seconds). + +**void JslSetTouchCallback(void(\*callback)(int, TOUCH\_STATE, TOUCH\_STATE, float))** - Set a callback function by which JoyShockLibrary can report the current touchpad state for each device. Only DualShock 4s will use this. This callback will be given the *deviceId* for the reporting device, its current and previous touchpad states, and the amount of time since the last report for this device (in seconds). + +**void JslSetConnectCallback(void(\*callback)(int))** - Set a callback function when a new device has been connected. This is *not* watching for new connections. You still need to explicitly call JslConnectDevices. From there, this callback will be called for each *new* device that is connected, giving you the opportunity to set default settings on those devices. It gives the unique deviceId for the newly connected device. This deviceId may be re-used if the device is connecting through the same port as a previous device. + +**void JslSetDisconnectCallback(void(\*callback)(int, bool))** - Set a callback function when a device has been disconnected. When this is called, the given deviceId is no longer useful to you -- all JslGet* functions will give empty/neutral results as there's no device at that id to read from. The bool your callback receives is whether or not this disconnect was due to a timeout. Otherwise, it's an explicit disconnection (device physically disconnected). + +**JSL_SETTINGS JslGetControllerInfoAndSettings(int deviceId)** - Read a bunch of info from the controller in one go instead of requesting each thing one at a time. Gyro space, colour (for LED on PlayStation controllers, for the controller itself for Switch controllers), whether it's calibrating, etc. See JoyShockLibrary.h for more info. + +**int JslGetControllerType(int deviceId)** - What type of controller is this device? + 1. Left Joy-Con + 2. Right Joy-Con + 3. Switch Pro Controller + 4. DualShock 4 + 5. DualSense + +**int JslGetControllerSplitType(int deviceId)** - Is this a half-controller or full? If half, what kind? + 1. Left half + 2. Right half + 3. Full controller + +**int JslGetControllerColour(int deviceId)** - Get the colour of the controller. Only Nintendo devices support this. Others will report white. + +**void JslSetLightColour(int deviceId, int colour)** - Set the light colour on the given controller. Only DualShock 4 and the DualSense support this. Players will often prefer to be able to disable the light, so make sure to give them that option, but when setting players up in a local multiplayer game, setting the light colour is a useful way to uniquely identify different controllers. + +**void JslSetPlayerNumber(int deviceId, int number)** - Set the lights that indicate player number. This only works on Nintendo devices and the DualSense. NOTE: The DualSense sets each LED through a bitmask. Use the ```DS5_PLAYER_#``` definitions in the header file to get PS5-style lightbar formats. + +Player 1: ```--x--``` +Player 2: ```-x-x-``` +Player 3: ```x-x-x``` +Player 4: ```xx-xx``` +Player 5: ```xxxxx``` + +**void JslSetRumble(int deviceId, int smallRumble, int bigRumble)** - DualShock 4s have two types of rumble, and they can be set at the same time with different intensities. These can be set from 0 to 255. Nintendo devices support rumble as well, but totally differently. They call it "HD rumble", and it's a great feature, but JoyShockLibrary doesn't yet support it. + +## Known and Perceived Issues +### Bluetooth connectivity +Joy-Cons and Pro Controllers are normally only be connected by Bluetooth. Some Bluetooth adapters can't keep up with these devices, resulting in laggy input. This is especially common when more than one device is connected (such as when using a pair of Joy-Cons). There is nothing JoyShockMapper or JoyShockLibrary can do about this. + +There is experimental support for connecting supported Switch controllers by USB now. Please let me know if you have any issues (or success!) with it. + +### Gyro poll rate on Nintendo devices +The Nintendo devices report every 15ms, but their IMUs actually report every 5ms. Every 15ms report includes the last 3 gyro and accelerometer reports. When creating the latest IMU state for Nintendo devices, JoyShockLibrary averages out those 3 gyro and accelerometer reports, so that it can best include all that information in a sensible format. For things like controlling a cursor on a plane, this should be of little to no consequence, since the result is the same as adding all 3 reports separately over shorter time intervals. But for representing real 3D rotations of the controller, this causes the Nintendo devices to be *slightly* less accurate than they could be, because we're combining 3 rotations in a simplistic way. + +In a future version I hope to either combine the 3 rotations in a way that works better in 3D, or to add a way for a single controller event to report several IMU events at the same time. + +## Backwards Compatibility +JoyShockLibrary v2 changes the gyro and accelerometer axes from previous versions. Previous versions were inconsistent between gyro and accelerometer. When upgrading to JoyShockLibrary v2, in order to maintain previous behaviour: +* Invert Gyro X +* Swap Accel Z and Y +* Then invert Accel Z + +JoyShockLibrary v3 makes some small changes to the behaviour of some functions: +* **JslConnectDevices** no longer calls **JslDisconnectAndDisposeAll** before looking for connections. Instead of reconnecting all devices, it'll only make new connections, without disrupting current connections. If you were counting on the old behaviour, call JslDisconnectAndDisposeAll before calling JslConnectDevices. +* Using a newer version of GamepadMotionHelpers, if you enable auto calibration, it will be applied more quickly at first, and then less quickly when it has more confidence in the current calibration. You can reset these at any time. + +## Credits +I'm Jibb Smart, and I made JoyShockLibrary. JoyShockLibrary has also benefited from the contributions of: +* Romeo Calota (Linux support + general portability improvements) +* RollinBarrel (touchpad support) +* Robin (wireless DS4/5 support) +* And others + +JoyShockLibrary uses substantial portions of mfosse's [JoyCon-Driver](https://github.com/mfosse/JoyCon-Driver), a [vJoy](http://vjoystick.sourceforge.net/site/) feeder for most communication with Nintendo devices, building on it with info from dekuNukem's [Nintendo Switch Reverse Engineering](https://github.com/dekuNukem/Nintendo_Switch_Reverse_Engineering/) page in order to (for example) unpack all gyro and accelerometer samples from each report. + +JoyShockLibrary's DualShock 4 support would likely not be possible without the info available on [PSDevWiki](https://www.psdevwiki.com/ps4/Main_Page) and [Eleccelerator Wiki](http://eleccelerator.com/wiki/index.php?title=DualShock_4). chrippa's [ds4drv](https://github.com/chrippa/ds4drv) was also a handy reference for getting rumble and lights working right away, and some changes have been made while referencing Ryochan7's [DS4Windows](https://github.com/Ryochan7/DS4Windows). + +This software depends on signal11's [HIDAPI](https://github.com/signal11/hidapi) to connect to USB and Bluetooth devices. + +The gravity calculation and gyro calibration is handled by another library of mine, [GamepadMotionHelpers](https://github.com/jibbsmart/gamepadmotionhelpers). Making it a separate library means you can make use of its robust sensor fusion calculation and automatic calibration options regardless of what you're using to read from the controller itself. + +## Helpful Resources +* [GyroWiki](http://gyrowiki.jibbsmart.com) - All about good gyro controls for games: + * Why gyro controls make gaming better; + * How developers can do a better job implementing gyro controls; + * How to use JoyShockLibrary; + * How gamers can play any PC game with gyro controls using [JoyShockMapper](https://github.com/Electronicks/JoyShockMapper). Legacy versions use JoyShockLibrary to read from supported controllers, but the standard version uses SDL2 to support more controllers. + +## License +JoyShockLibrary is licensed under the MIT License - see [LICENSE.md](LICENSE.md). diff --git a/Memoria.XInputDotNetPure/JoyShockLibrary/x64/JoyShockLibrary.dll b/Memoria.XInputDotNetPure/JoyShockLibrary/x64/JoyShockLibrary.dll new file mode 100644 index 000000000..4c0237d72 Binary files /dev/null and b/Memoria.XInputDotNetPure/JoyShockLibrary/x64/JoyShockLibrary.dll differ diff --git a/Memoria.XInputDotNetPure/JoyShockLibrary/x86/JoyShockLibrary.dll b/Memoria.XInputDotNetPure/JoyShockLibrary/x86/JoyShockLibrary.dll new file mode 100644 index 000000000..53266b57d Binary files /dev/null and b/Memoria.XInputDotNetPure/JoyShockLibrary/x86/JoyShockLibrary.dll differ diff --git a/Memoria.XInputDotNetPure/Memoria.XInputDotNetPure.csproj b/Memoria.XInputDotNetPure/Memoria.XInputDotNetPure.csproj new file mode 100644 index 000000000..bbe7ec6f6 --- /dev/null +++ b/Memoria.XInputDotNetPure/Memoria.XInputDotNetPure.csproj @@ -0,0 +1,69 @@ + + + + Debug + AnyCPU + 9.0.30729 + 2.0 + {952700CD-A384-43A4-AC4B-52256F83836E} + Library + Properties + XInputDotNetPure + XInputDotNetPure + v3.5 + 512 + Unity Full v3.5 + + + true + ..\Output\ + TRACE;DEBUG;WIN64 + full + AnyCPU + 7.3 + prompt + false + false + + + true + ..\Output\ + TRACE + true + portable + AnyCPU + 7.3 + prompt + false + false + + + + + + + + + + + + + + + + + PreserveNewest + + + PreserveNewest + + + + + \ No newline at end of file diff --git a/Memoria.XInputDotNetPure/Properties/AssemblyInfo.cs b/Memoria.XInputDotNetPure/Properties/AssemblyInfo.cs new file mode 100644 index 000000000..3027781db --- /dev/null +++ b/Memoria.XInputDotNetPure/Properties/AssemblyInfo.cs @@ -0,0 +1,36 @@ +using System.Reflection; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; + +// General Information about an assembly is controlled through the following +// set of attributes. Change these attribute values to modify the information +// associated with an assembly. +[assembly: AssemblyTitle("XInputDotNetPure")] +[assembly: AssemblyDescription("")] +[assembly: AssemblyConfiguration("")] +[assembly: AssemblyCompany("")] +[assembly: AssemblyProduct("Memoria.XInputDotNetPure")] +[assembly: AssemblyCopyright("Copyright © 2009")] +[assembly: AssemblyTrademark("")] +[assembly: AssemblyCulture("")] + +// Setting ComVisible to false makes the types in this assembly not visible +// to COM components. If you need to access a type in this assembly from +// COM, set the ComVisible attribute to true on that type. +[assembly: ComVisible(false)] + +// The following GUID is for the ID of the typelib if this project is exposed to COM +[assembly: Guid("1fee853a-0e42-4b62-8463-910b7ec4f726")] + +// Version information for an assembly consists of the following four values: +// +// Major Version +// Minor Version +// Build Number +// Revision +// +// You can specify all the values or you can default the Build and Revision Numbers +// by using the '*' as shown below: +// [assembly: AssemblyVersion("1.0.*")] +[assembly: AssemblyVersion("1.0.0.0")] +[assembly: AssemblyFileVersion("1.0.0.0")] diff --git a/Memoria.XInputDotNetPure/Utils.cs b/Memoria.XInputDotNetPure/Utils.cs new file mode 100644 index 000000000..98ee3daf9 --- /dev/null +++ b/Memoria.XInputDotNetPure/Utils.cs @@ -0,0 +1,97 @@ +using System; + +namespace XInputDotNetPure +{ + static class Utils + { + public const uint Success = 0x000; + public const uint NotConnected = 0x000; + + private const int LeftStickDeadZone = 7849; + private const int RightStickDeadZone = 8689; + private const int TriggerDeadZone = 30; + + public static float ApplyTriggerDeadZone(byte value, GamePadDeadZone deadZoneMode) + { + if (deadZoneMode == GamePadDeadZone.None) + { + return ApplyDeadZone(value, byte.MaxValue, 0.0f); + } + else + { + return ApplyDeadZone(value, byte.MaxValue, TriggerDeadZone); + } + } + + public static GamePadThumbSticks.StickValue ApplyLeftStickDeadZone(short valueX, short valueY, GamePadDeadZone deadZoneMode) + { + return ApplyStickDeadZone(valueX, valueY, deadZoneMode, LeftStickDeadZone); + } + + public static GamePadThumbSticks.StickValue ApplyRightStickDeadZone(short valueX, short valueY, GamePadDeadZone deadZoneMode) + { + return ApplyStickDeadZone(valueX, valueY, deadZoneMode, RightStickDeadZone); + } + + private static GamePadThumbSticks.StickValue ApplyStickDeadZone(short valueX, short valueY, GamePadDeadZone deadZoneMode, int deadZoneSize) + { + if (deadZoneMode == GamePadDeadZone.Circular) + { + // Cast to long to avoid int overflow if valueX and valueY are both 32768, which would result in a negative number and Sqrt returns NaN + float distanceFromCenter = (float)Math.Sqrt((long)valueX * (long)valueX + (long)valueY * (long)valueY); + float coefficient = ApplyDeadZone(distanceFromCenter, short.MaxValue, deadZoneSize); + coefficient = coefficient > 0.0f ? coefficient / distanceFromCenter : 0.0f; + return new GamePadThumbSticks.StickValue( + Clamp(valueX * coefficient), + Clamp(valueY * coefficient) + ); + } + else if (deadZoneMode == GamePadDeadZone.IndependentAxes) + { + return new GamePadThumbSticks.StickValue( + ApplyDeadZone(valueX, short.MaxValue, deadZoneSize), + ApplyDeadZone(valueY, short.MaxValue, deadZoneSize) + ); + } + else + { + return new GamePadThumbSticks.StickValue( + ApplyDeadZone(valueX, short.MaxValue, 0.0f), + ApplyDeadZone(valueY, short.MaxValue, 0.0f) + ); + } + } + + public static T Clamp(this T val, T min, T max) where T : IComparable + { + if (val.CompareTo(min) < 0) return min; + else if (val.CompareTo(max) > 0) return max; + else return val; + } + + private static float Clamp(float value) + { + return value < -1.0f ? -1.0f : (value > 1.0f ? 1.0f : value); + } + + private static float ApplyDeadZone(float value, float maxValue, float deadZoneSize) + { + if (value < -deadZoneSize) + { + value += deadZoneSize; + } + else if (value > deadZoneSize) + { + value -= deadZoneSize; + } + else + { + return 0.0f; + } + + value /= maxValue - deadZoneSize; + + return Clamp(value); + } + } +} diff --git a/Memoria.sln b/Memoria.sln index 18397348b..0a1eb7440 100644 --- a/Memoria.sln +++ b/Memoria.sln @@ -74,6 +74,8 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Utils", "Utils", "{9A4F68D4 EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Memoria.SteamFix", "Memoria.SteamFix\Memoria.SteamFix.csproj", "{8E82C124-FFC6-4B99-B5A9-F65B657E3822}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Memoria.XInputDotNetPure", "Memoria.XInputDotNetPure\Memoria.XInputDotNetPure.csproj", "{952700CD-A384-43A4-AC4B-52256F83836E}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -222,6 +224,18 @@ Global {8E82C124-FFC6-4B99-B5A9-F65B657E3822}.Release|x64.Build.0 = Release|Any CPU {8E82C124-FFC6-4B99-B5A9-F65B657E3822}.Release|x86.ActiveCfg = Release|Any CPU {8E82C124-FFC6-4B99-B5A9-F65B657E3822}.Release|x86.Build.0 = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|Any CPU.Build.0 = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|x64.ActiveCfg = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|x64.Build.0 = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|x86.ActiveCfg = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Debug|x86.Build.0 = Debug|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|Any CPU.ActiveCfg = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|Any CPU.Build.0 = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|x64.ActiveCfg = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|x64.Build.0 = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|x86.ActiveCfg = Release|Any CPU + {952700CD-A384-43A4-AC4B-52256F83836E}.Release|x86.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE diff --git a/References/Memoria.MSBuild.dll b/References/Memoria.MSBuild.dll index 1c471daa5..53493b690 100644 Binary files a/References/Memoria.MSBuild.dll and b/References/Memoria.MSBuild.dll differ