using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using StardewModdingAPI.Toolkit.Serialization;
namespace StardewModdingAPI.Utilities
{
/// A set of multi-key bindings which can be triggered by the player.
public class KeybindList
{
/*********
** Accessors
*********/
/// The individual keybinds.
public Keybind[] Keybinds { get; }
/// Whether any keys are bound.
public bool IsBound { get; }
/*********
** Public methods
*********/
/// Construct an instance.
/// The underlying keybinds.
/// See or to parse it from a string representation. You can also use this type directly in your config or JSON data models, and it'll be parsed by SMAPI.
public KeybindList(params Keybind[] keybinds)
{
this.Keybinds = keybinds.Where(p => p.IsBound).ToArray();
this.IsBound = this.Keybinds.Any();
}
/// Construct an instance.
/// A single-key binding.
public KeybindList(SButton singleKey)
: this(new Keybind(singleKey)) { }
/// Parse a keybind list from a string, and throw an exception if it's not valid.
/// The keybind string. See remarks on for format details.
/// The format is invalid.
public static KeybindList Parse(string input)
{
return KeybindList.TryParse(input, out KeybindList? parsed, out string[] errors)
? parsed
: throw new SParseException($"Can't parse {nameof(Keybind)} from invalid value '{input}'.\n{string.Join("\n", errors)}");
}
/// Try to parse a keybind list from a string.
/// The keybind string. See remarks on for format details.
/// The parsed keybind list, if valid.
/// The errors that occurred while parsing the input, if any.
public static bool TryParse(string input, [NotNullWhen(true)] out KeybindList? parsed, out string[] errors)
{
// empty input
if (string.IsNullOrWhiteSpace(input))
{
parsed = new KeybindList();
errors = Array.Empty();
return true;
}
// parse buttons
var rawErrors = new List();
var keybinds = new List();
foreach (string rawSet in input.Split(','))
{
if (string.IsNullOrWhiteSpace(rawSet))
continue;
if (!Keybind.TryParse(rawSet, out Keybind? keybind, out string[] curErrors))
rawErrors.AddRange(curErrors);
else
keybinds.Add(keybind);
}
// build result
if (rawErrors.Any())
{
parsed = null;
errors = rawErrors.Distinct().ToArray();
return false;
}
else
{
parsed = new KeybindList(keybinds.ToArray());
errors = Array.Empty();
return true;
}
}
/// Get a keybind list for a single keybind.
/// The buttons that must be down to activate the keybind.
public static KeybindList ForSingle(params SButton[] buttons)
{
return new KeybindList(
new Keybind(buttons)
);
}
/// Get the overall keybind list state relative to the previous tick.
/// States are transitive across keybind. For example, if one keybind is 'released' and another is 'pressed', the state of the keybind list is 'held'.
public SButtonState GetState()
{
bool wasPressed = false;
bool isPressed = false;
foreach (Keybind keybind in this.Keybinds)
{
switch (keybind.GetState())
{
case SButtonState.Pressed:
isPressed = true;
break;
case SButtonState.Held:
wasPressed = true;
isPressed = true;
break;
case SButtonState.Released:
wasPressed = true;
break;
}
}
if (wasPressed == isPressed)
{
return wasPressed
? SButtonState.Held
: SButtonState.None;
}
return wasPressed
? SButtonState.Released
: SButtonState.Pressed;
}
/// Get whether any of the button sets are pressed.
public bool IsDown()
{
SButtonState state = this.GetState();
return state is SButtonState.Pressed or SButtonState.Held;
}
/// Get whether the input binding was just pressed this tick.
public bool JustPressed()
{
return this.GetState() == SButtonState.Pressed;
}
/// Get the keybind which is currently down, if any. If there are multiple keybinds down, the first one is returned.
public Keybind? GetKeybindCurrentlyDown()
{
return this.Keybinds.FirstOrDefault(p => p.GetState().IsDown());
}
/// Get a string representation of the input binding.
/// A keybind list is serialized to a string like LeftControl + S, LeftAlt + S, where each multi-key binding is separated with , and the keys within each keybind are separated with +. The key order is commutative, so LeftControl + S and S + LeftControl are identical.
public override string ToString()
{
return this.Keybinds.Length > 0
? string.Join(", ", this.Keybinds.Select(p => p.ToString()))
: SButton.None.ToString();
}
}
}