forked from Mirror/Ryujinx
da4e0856c9
* controller image changes depending on the selected controller type the new controller image assets are temporary until i get new ones * Game list scans subdirs for games * Key file existence check * Only shows Program NCAs in Application list * Change shown GUI columns without restarting * Sort by column if you click on the column header Columns are sorted as text so there are inaccuracies on some columns * Fix sort on Time Played, Last Played and File Size columns * Add ability to designate favourite games #1 TODO: - Make fav games persistent - Fix invisible check marks due to theme * Add ability to designate favourite games #2 Also removed default theme * Added a Windows specific build condition and a Linux bug fix * bugfix * Load metadata from JSONs * Temp bug fix for MacOS * lil clean up * requested changes * Misc fixes * edited schema and config * Show the TitleID of games on the title bar * gui column config option have names * Async loading of game list * bugfix and cleanup * thog's requested changes * requested changes and cleanup still need to fix the gtk seizure * Fix issue where an ExeFS as a NSP didn't show up in the application list * Minor fixes * catch glib unhandled exceptions * Make sure to do UI manipulation in the main thread * Print path of invalid files * Ac_k's requested changes * Return of the dark theme * move AboutInfo struct to another file * sort usings * changes - gdkchan's requested changes that have been marked resolved - made some structs internal as they aren't used outside of the GUI - renamed Ryujinx.UI to Ryujinx.Ui to fit naming convention and folder structure - fixed bug where controller type dropdown box is stretched
246 lines
7.8 KiB
C#
246 lines
7.8 KiB
C#
using OpenTK;
|
|
using OpenTK.Input;
|
|
using Ryujinx.HLE.Input;
|
|
using System;
|
|
|
|
namespace Ryujinx.Ui.Input
|
|
{
|
|
public enum ControllerInputId
|
|
{
|
|
Button0,
|
|
Button1,
|
|
Button2,
|
|
Button3,
|
|
Button4,
|
|
Button5,
|
|
Button6,
|
|
Button7,
|
|
Button8,
|
|
Button9,
|
|
Button10,
|
|
Button11,
|
|
Button12,
|
|
Button13,
|
|
Button14,
|
|
Button15,
|
|
Button16,
|
|
Button17,
|
|
Button18,
|
|
Button19,
|
|
Button20,
|
|
Axis0,
|
|
Axis1,
|
|
Axis2,
|
|
Axis3,
|
|
Axis4,
|
|
Axis5,
|
|
Hat0Up,
|
|
Hat0Down,
|
|
Hat0Left,
|
|
Hat0Right,
|
|
Hat1Up,
|
|
Hat1Down,
|
|
Hat1Left,
|
|
Hat1Right,
|
|
Hat2Up,
|
|
Hat2Down,
|
|
Hat2Left,
|
|
Hat2Right,
|
|
}
|
|
|
|
public struct NpadControllerLeft
|
|
{
|
|
public ControllerInputId Stick;
|
|
public ControllerInputId StickButton;
|
|
public ControllerInputId ButtonMinus;
|
|
public ControllerInputId ButtonL;
|
|
public ControllerInputId ButtonZl;
|
|
public ControllerInputId DPadUp;
|
|
public ControllerInputId DPadDown;
|
|
public ControllerInputId DPadLeft;
|
|
public ControllerInputId DPadRight;
|
|
}
|
|
|
|
public struct NpadControllerRight
|
|
{
|
|
public ControllerInputId Stick;
|
|
public ControllerInputId StickButton;
|
|
public ControllerInputId ButtonA;
|
|
public ControllerInputId ButtonB;
|
|
public ControllerInputId ButtonX;
|
|
public ControllerInputId ButtonY;
|
|
public ControllerInputId ButtonPlus;
|
|
public ControllerInputId ButtonR;
|
|
public ControllerInputId ButtonZr;
|
|
}
|
|
|
|
public class NpadController
|
|
{
|
|
/// <summary>
|
|
/// Enables or disables controller support
|
|
/// </summary>
|
|
public bool Enabled { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Controller Device Index
|
|
/// </summary>
|
|
public int Index { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Controller Analog Stick Deadzone
|
|
/// </summary>
|
|
public float Deadzone { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Controller Trigger Threshold
|
|
/// </summary>
|
|
public float TriggerThreshold { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Left JoyCon Controller Bindings
|
|
/// </summary>
|
|
public NpadControllerLeft LeftJoycon { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Right JoyCon Controller Bindings
|
|
/// </summary>
|
|
public NpadControllerRight RightJoycon { get; private set; }
|
|
|
|
public NpadController(
|
|
bool enabled,
|
|
int index,
|
|
float deadzone,
|
|
float triggerThreshold,
|
|
NpadControllerLeft leftJoycon,
|
|
NpadControllerRight rightJoycon)
|
|
{
|
|
Enabled = enabled;
|
|
Index = index;
|
|
Deadzone = deadzone;
|
|
TriggerThreshold = triggerThreshold;
|
|
LeftJoycon = leftJoycon;
|
|
RightJoycon = rightJoycon;
|
|
}
|
|
|
|
public void SetEnabled(bool enabled)
|
|
{
|
|
Enabled = enabled;
|
|
}
|
|
|
|
public ControllerButtons GetButtons()
|
|
{
|
|
if (!Enabled)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
JoystickState joystickState = Joystick.GetState(Index);
|
|
|
|
ControllerButtons buttons = 0;
|
|
|
|
if (IsActivated(joystickState, LeftJoycon.DPadUp)) buttons |= ControllerButtons.DpadUp;
|
|
if (IsActivated(joystickState, LeftJoycon.DPadDown)) buttons |= ControllerButtons.DpadDown;
|
|
if (IsActivated(joystickState, LeftJoycon.DPadLeft)) buttons |= ControllerButtons.DpadLeft;
|
|
if (IsActivated(joystickState, LeftJoycon.DPadRight)) buttons |= ControllerButtons.DPadRight;
|
|
if (IsActivated(joystickState, LeftJoycon.StickButton)) buttons |= ControllerButtons.StickLeft;
|
|
if (IsActivated(joystickState, LeftJoycon.ButtonMinus)) buttons |= ControllerButtons.Minus;
|
|
if (IsActivated(joystickState, LeftJoycon.ButtonL)) buttons |= ControllerButtons.L;
|
|
if (IsActivated(joystickState, LeftJoycon.ButtonZl)) buttons |= ControllerButtons.Zl;
|
|
|
|
if (IsActivated(joystickState, RightJoycon.ButtonA)) buttons |= ControllerButtons.A;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonB)) buttons |= ControllerButtons.B;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonX)) buttons |= ControllerButtons.X;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonY)) buttons |= ControllerButtons.Y;
|
|
if (IsActivated(joystickState, RightJoycon.StickButton)) buttons |= ControllerButtons.StickRight;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonPlus)) buttons |= ControllerButtons.Plus;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonR)) buttons |= ControllerButtons.R;
|
|
if (IsActivated(joystickState, RightJoycon.ButtonZr)) buttons |= ControllerButtons.Zr;
|
|
|
|
return buttons;
|
|
}
|
|
|
|
private bool IsActivated(JoystickState joystickState,ControllerInputId controllerInputId)
|
|
{
|
|
if (controllerInputId <= ControllerInputId.Button20)
|
|
{
|
|
return joystickState.IsButtonDown((int)controllerInputId);
|
|
}
|
|
else if (controllerInputId <= ControllerInputId.Axis5)
|
|
{
|
|
int axis = controllerInputId - ControllerInputId.Axis0;
|
|
|
|
return joystickState.GetAxis(axis) > TriggerThreshold;
|
|
}
|
|
else if (controllerInputId <= ControllerInputId.Hat2Right)
|
|
{
|
|
int hat = (controllerInputId - ControllerInputId.Hat0Up) / 4;
|
|
|
|
int baseHatId = (int)ControllerInputId.Hat0Up + (hat * 4);
|
|
|
|
JoystickHatState hatState = joystickState.GetHat((JoystickHat)hat);
|
|
|
|
if (hatState.IsUp && ((int)controllerInputId % baseHatId == 0)) return true;
|
|
if (hatState.IsDown && ((int)controllerInputId % baseHatId == 1)) return true;
|
|
if (hatState.IsLeft && ((int)controllerInputId % baseHatId == 2)) return true;
|
|
if (hatState.IsRight && ((int)controllerInputId % baseHatId == 3)) return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public (short, short) GetLeftStick()
|
|
{
|
|
if (!Enabled)
|
|
{
|
|
return (0, 0);
|
|
}
|
|
|
|
return GetStick(LeftJoycon.Stick);
|
|
}
|
|
|
|
public (short, short) GetRightStick()
|
|
{
|
|
if (!Enabled)
|
|
{
|
|
return (0, 0);
|
|
}
|
|
|
|
return GetStick(RightJoycon.Stick);
|
|
}
|
|
|
|
private (short, short) GetStick(ControllerInputId stickInputId)
|
|
{
|
|
if (stickInputId < ControllerInputId.Axis0 || stickInputId > ControllerInputId.Axis5)
|
|
{
|
|
return (0, 0);
|
|
}
|
|
|
|
JoystickState jsState = Joystick.GetState(Index);
|
|
|
|
int xAxis = stickInputId - ControllerInputId.Axis0;
|
|
|
|
float xValue = jsState.GetAxis(xAxis);
|
|
float yValue = 0 - jsState.GetAxis(xAxis + 1); // Invert Y-axis
|
|
|
|
return ApplyDeadzone(new Vector2(xValue, yValue));
|
|
}
|
|
|
|
private (short, short) ApplyDeadzone(Vector2 axis)
|
|
{
|
|
return (ClampAxis(MathF.Abs(axis.X) > Deadzone ? axis.X : 0f),
|
|
ClampAxis(MathF.Abs(axis.Y) > Deadzone ? axis.Y : 0f));
|
|
}
|
|
|
|
private static short ClampAxis(float value)
|
|
{
|
|
if (value <= -short.MaxValue)
|
|
{
|
|
return -short.MaxValue;
|
|
}
|
|
else
|
|
{
|
|
return (short)(value * short.MaxValue);
|
|
}
|
|
}
|
|
}
|
|
}
|