mirror of
https://github.com/Artemis-RGB/Artemis
synced 2025-12-13 05:48:35 +00:00
160 lines
5.0 KiB
C#
160 lines
5.0 KiB
C#
using System;
|
|
using System.Linq;
|
|
using Artemis.Core;
|
|
using Artemis.Core.Services;
|
|
using Avalonia;
|
|
using Avalonia.Controls;
|
|
using Avalonia.Data;
|
|
using Avalonia.Input;
|
|
using Avalonia.Interactivity;
|
|
using Avalonia.Markup.Xaml;
|
|
using Avalonia.Threading;
|
|
using DryIoc;
|
|
using FluentAvalonia.Core;
|
|
using Humanizer;
|
|
using Material.Icons;
|
|
|
|
namespace Artemis.UI.Shared;
|
|
|
|
/// <summary>
|
|
/// Represents a control that can be used to display or edit <see cref="Core.Hotkey" /> instances.
|
|
/// </summary>
|
|
public partial class HotkeyBox : UserControl
|
|
{
|
|
private readonly IInputService _inputService;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the <see cref="HotkeyBox" /> class
|
|
/// </summary>
|
|
public HotkeyBox()
|
|
{
|
|
_inputService = UI.Locator.Resolve<IInputService>();
|
|
|
|
InitializeComponent();
|
|
PropertyChanged += OnPropertyChanged;
|
|
UpdateDisplayTextBox();
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void OnGotFocus(GotFocusEventArgs e)
|
|
{
|
|
_inputService.KeyboardKeyDown += InputServiceOnKeyboardKeyDown;
|
|
_inputService.KeyboardKeyUp += InputServiceOnKeyboardKeyUp;
|
|
|
|
base.OnGotFocus(e);
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void OnLostFocus(RoutedEventArgs e)
|
|
{
|
|
_inputService.KeyboardKeyDown -= InputServiceOnKeyboardKeyDown;
|
|
_inputService.KeyboardKeyUp -= InputServiceOnKeyboardKeyUp;
|
|
|
|
base.OnLostFocus(e);
|
|
}
|
|
|
|
private void OnPropertyChanged(object? sender, AvaloniaPropertyChangedEventArgs e)
|
|
{
|
|
if (e.Property == HotkeyProperty)
|
|
UpdateDisplayTextBox();
|
|
}
|
|
|
|
private void InputServiceOnKeyboardKeyDown(object? sender, ArtemisKeyboardKeyEventArgs e)
|
|
{
|
|
if (e.Key >= KeyboardKey.LeftShift && e.Key <= KeyboardKey.RightAlt)
|
|
return;
|
|
|
|
Hotkey ??= new Hotkey();
|
|
Hotkey.Key = e.Key;
|
|
Hotkey.Modifiers = e.Modifiers;
|
|
|
|
Dispatcher.UIThread.Post(() =>
|
|
{
|
|
UpdateDisplayTextBox();
|
|
HotkeyChanged?.Invoke(this, EventArgs.Empty);
|
|
});
|
|
}
|
|
|
|
private void InputServiceOnKeyboardKeyUp(object? sender, ArtemisKeyboardKeyEventArgs e)
|
|
{
|
|
if (e.Modifiers == KeyboardModifierKey.None)
|
|
Dispatcher.UIThread.Post(() => FocusManager.Instance?.Focus(null));
|
|
}
|
|
|
|
private void UpdateDisplayTextBox()
|
|
{
|
|
string? display = null;
|
|
if (Hotkey?.Modifiers != null)
|
|
display = string.Join("+", Enum.GetValues<KeyboardModifierKey>().Skip(1).Where(m => Hotkey.Modifiers.Value.HasFlag(m)).Select(v => v.Humanize()));
|
|
if (Hotkey?.Key != null)
|
|
display = string.IsNullOrEmpty(display) ? Hotkey.Key.ToString() : $"{display}+{Hotkey.Key}";
|
|
|
|
DisplayTextBox.Text = display;
|
|
DisplayTextBox.CaretIndex = DisplayTextBox.Text?.Length ?? 0;
|
|
}
|
|
|
|
private void Button_OnClick(object? sender, RoutedEventArgs e)
|
|
{
|
|
Hotkey = null;
|
|
FocusManager.Instance?.Focus(null);
|
|
|
|
UpdateDisplayTextBox();
|
|
}
|
|
|
|
#region Properties
|
|
|
|
/// <summary>
|
|
/// Gets or sets the currently displayed icon as either a <see cref="MaterialIconKind" /> or an <see cref="Uri" />
|
|
/// pointing to an SVG
|
|
/// </summary>
|
|
public static readonly StyledProperty<Hotkey?> HotkeyProperty = AvaloniaProperty.Register<HotkeyBox, Hotkey?>(nameof(Hotkey), defaultBindingMode: BindingMode.TwoWay);
|
|
|
|
/// <summary>
|
|
/// Gets or sets the watermark of the hotkey box when it is empty.
|
|
/// </summary>
|
|
public static readonly StyledProperty<string?> WatermarkProperty = AvaloniaProperty.Register<HotkeyBox, string?>(nameof(Watermark));
|
|
|
|
/// <summary>
|
|
/// Gets or sets a boolean indicating whether the watermark should float above the hotkey box when it is not empty.
|
|
/// </summary>
|
|
public static readonly StyledProperty<bool> UseFloatingWatermarkProperty = AvaloniaProperty.Register<HotkeyBox, bool>(nameof(UseFloatingWatermark));
|
|
|
|
/// <summary>
|
|
/// Gets or sets the currently displayed icon as either a <see cref="MaterialIconKind" /> or an <see cref="Uri" />
|
|
/// pointing to an SVG
|
|
/// </summary>
|
|
public Hotkey? Hotkey
|
|
{
|
|
get => GetValue(HotkeyProperty);
|
|
set => SetValue(HotkeyProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the watermark of the hotkey box when it is empty.
|
|
/// </summary>
|
|
public string? Watermark
|
|
{
|
|
get => GetValue(WatermarkProperty);
|
|
set => SetValue(WatermarkProperty, value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets a boolean indicating whether the watermark should float above the hotkey box when it is not empty.
|
|
/// </summary>
|
|
public bool UseFloatingWatermark
|
|
{
|
|
get => GetValue(UseFloatingWatermarkProperty);
|
|
set => SetValue(UseFloatingWatermarkProperty, value);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Events
|
|
|
|
/// <summary>
|
|
/// Occurs when the hotkey changes.
|
|
/// </summary>
|
|
public event TypedEventHandler<HotkeyBox, EventArgs>? HotkeyChanged;
|
|
|
|
#endregion
|
|
} |