diff --git a/BizHawk.Client.Common/Api/Classes/JoypadApi.cs b/BizHawk.Client.Common/Api/Classes/JoypadApi.cs index 79dce1ff00..0671cd82c2 100644 --- a/BizHawk.Client.Common/Api/Classes/JoypadApi.cs +++ b/BizHawk.Client.Common/Api/Classes/JoypadApi.cs @@ -44,7 +44,9 @@ namespace BizHawk.Client.Common public void Set(Dictionary buttons, int? controller = null) { - foreach (var button in Global.InputManager.ActiveController.Definition.BoolButtons) + // If a controller is specified, we need to iterate over unique button names. If not, we iterate over + // ALL button names with P{controller} prefixes + foreach (var button in Global.InputManager.ActiveController.ToBoolButtonNameList(controller)) { Set(button, buttons.TryGetValue(button, out var state) ? state : (bool?) null, controller); } diff --git a/BizHawk.Client.Common/lua/EmuLuaLibrary.Joypad.cs b/BizHawk.Client.Common/lua/EmuLuaLibrary.Joypad.cs index c94b150020..12990100f4 100644 --- a/BizHawk.Client.Common/lua/EmuLuaLibrary.Joypad.cs +++ b/BizHawk.Client.Common/lua/EmuLuaLibrary.Joypad.cs @@ -45,7 +45,7 @@ namespace BizHawk.Client.Common public void Set(LuaTable buttons, int? controller = null) { var dict = new Dictionary(); - foreach (var k in buttons.Keys) dict[k.ToString()] = (bool) buttons[k]; + foreach (var k in buttons.Keys) dict[k.ToString()] = Convert.ToBoolean(buttons[k]); // Accepts 1/0 or true/false APIs.Joypad.Set(dict, controller); } diff --git a/BizHawk.Emulation.Common/Extensions.cs b/BizHawk.Emulation.Common/Extensions.cs index dcf2998b2c..3f70690cee 100644 --- a/BizHawk.Emulation.Common/Extensions.cs +++ b/BizHawk.Emulation.Common/Extensions.cs @@ -314,6 +314,47 @@ namespace BizHawk.Emulation.Common return !info.GetCustomAttributes(false).Any(a => a is FeatureNotImplementedAttribute); } + /// + /// Gets a list of boolean button names. If a controller number is specified, only returns button names + /// (without the "P" prefix) that match that controller number. If a controller number is NOT specified, + /// then all button names are returned. + /// + /// For example, consider example "P1 A", "P1 B", "P2 A", "P2 B". See below for sample outputs: + /// - ToBoolButtonNameList(controller, 1) -> [A, B] + /// - ToBoolButtonNameList(controller, 2) -> [A, B] + /// - ToBoolButtonNameList(controller, null) -> [P1 A, P1 B, P2 A, P2 B] + /// + public static List ToBoolButtonNameList(this IController controller, int? controllerNum = null) + { + return ToControlNameList(controller.Definition.BoolButtons, controllerNum); + } + + /// + /// See ToBoolButtonNameList(). Works the same except with float controls + /// + public static List ToFloatControlNameList(this IController controller, int? controllerNum = null) + { + return ToControlNameList(controller.Definition.FloatControls, controllerNum); + } + + private static List ToControlNameList(List buttonList, int? controllerNum = null) + { + var buttons = new List(); + foreach (var button in buttonList) + { + if (controllerNum != null && button.Length > 2 && button.Substring(0, 2) == $"P{controllerNum}") + { + var sub = button.Substring(3); + buttons.Add(sub); + } + else if (controllerNum == null) + { + buttons.Add(button); + } + } + return buttons; + } + public static IDictionary ToDictionary(this IController controller, int? controllerNum = null) { var buttons = new Dictionary();