SystemTrayMenu/Business/KeyboardInput.cs

634 lines
23 KiB
C#
Raw Normal View History

// <copyright file="KeyboardInput.cs" company="PlaceholderCompany">
// Copyright (c) PlaceholderCompany. All rights reserved.
// </copyright>
namespace SystemTrayMenu.Handler
{
using System;
using System.Globalization;
using System.Linq;
using System.Windows.Controls;
using System.Windows.Input;
using SystemTrayMenu.DataClasses;
using SystemTrayMenu.Helpers;
using SystemTrayMenu.Utilities;
using static SystemTrayMenu.UserInterface.Menu;
using Menu = SystemTrayMenu.UserInterface.Menu;
internal class KeyboardInput : IDisposable
{
private readonly Menu?[] menus;
private readonly KeyboardHook hook = new();
private Menu? focussedMenu;
private ListViewItemData? focussedRow;
public KeyboardInput(Menu?[] menus)
{
this.menus = menus;
}
internal event Action? HotKeyPressed;
internal event Action? ClosePressed;
internal event Action<ListView, ListViewItemData>? RowSelected;
2023-04-17 09:27:27 +12:00
internal event Action<int, ListView?>? RowDeselected;
internal event Action<ListView, ListViewItemData>? EnterPressed;
internal bool InUse { get; set; }
public void Dispose()
{
hook.Dispose();
}
internal void RegisterHotKey()
{
if (!string.IsNullOrEmpty(Properties.Settings.Default.HotKey))
{
try
{
hook.RegisterHotKey();
hook.KeyPressed += (sender, e) => HotKeyPressed?.Invoke();
}
catch (InvalidOperationException ex)
{
Log.Warn($"key:'{Properties.Settings.Default.HotKey}'", ex);
Properties.Settings.Default.HotKey = string.Empty;
Properties.Settings.Default.Save();
}
}
}
internal void ResetSelectedByKey()
{
focussedMenu = null;
focussedRow = null;
}
2023-05-03 08:04:32 +12:00
internal void CmdKeyProcessed(Menu sender, Key key, ModifierKeys modifiers)
{
2022-11-29 08:27:52 +13:00
switch (key)
{
case Key.Enter:
2022-11-29 08:27:52 +13:00
if (modifiers == ModifierKeys.None)
{
SelectByKey(key, modifiers);
focussedMenu?.FocusTextBox();
2022-11-29 08:27:52 +13:00
}
break;
case Key.Left:
case Key.Right:
case Key.Home:
case Key.End:
case Key.Up:
case Key.Down:
case Key.Escape:
2022-11-29 08:27:52 +13:00
if (modifiers == ModifierKeys.None)
{
SelectByKey(key, modifiers);
}
break;
2022-11-29 08:27:52 +13:00
case Key.F4:
if (modifiers == ModifierKeys.Alt)
{
SelectByKey(key, modifiers);
}
break;
case Key.F:
if (modifiers == ModifierKeys.Control)
{
focussedMenu?.FocusTextBox();
2022-11-29 08:27:52 +13:00
}
break;
case Key.Tab:
2022-11-29 08:27:52 +13:00
if (modifiers == ModifierKeys.None)
{
2022-11-29 08:27:52 +13:00
int indexOfTheCurrentMenu = GetMenuIndex(sender);
int indexMax = menus.Where(m => m != null).Count() - 1;
int indexNew = 0;
if (indexOfTheCurrentMenu > 0)
{
indexNew = indexOfTheCurrentMenu - 1;
}
else
{
indexNew = indexMax;
}
menus[indexNew]?.FocusTextBox();
}
2022-11-29 08:27:52 +13:00
else if (modifiers == ModifierKeys.Shift)
{
2022-11-29 08:27:52 +13:00
int indexOfTheCurrentMenu = GetMenuIndex(sender);
int indexMax = menus.Where(m => m != null).Count() - 1;
int indexNew = 0;
if (indexOfTheCurrentMenu < indexMax)
{
indexNew = indexOfTheCurrentMenu + 1;
}
else
{
indexNew = 0;
}
menus[indexNew]?.FocusTextBox();
}
break;
case Key.Apps:
2022-11-29 08:27:52 +13:00
if (modifiers == ModifierKeys.None)
{
ListView? dgv = focussedMenu?.GetDataGridView();
if (dgv != null)
{
if (focussedRow != null)
{
#if TODO // WPF: Better way to open context menu (as it looks like this is the code's intention)
Point point = dgv.GetCellDisplayRectangle(2, iRowKey, false).Location;
RowData trigger = (RowData)dgv.Rows[iRowKey].Cells[2].Value;
MouseEventArgs mouseEventArgs = new(MouseButtons.Right, 1, point.X, point.Y, 0);
trigger.MouseDown(dgv, mouseEventArgs);
#endif
}
}
}
break;
default:
break;
}
2023-04-17 09:27:27 +12:00
int GetMenuIndex(in Menu? currentMenu)
{
int index = 0;
2023-04-17 09:27:27 +12:00
foreach (Menu? menuFindIndex in menus.Where(m => m != null))
{
if (currentMenu == menuFindIndex)
{
break;
}
index++;
}
return index;
}
}
internal void SearchTextChanging()
{
ClearIsSelectedByKey();
}
internal void SearchTextChanged(Menu menu, bool isSearchStringEmpty)
{
if (isSearchStringEmpty)
{
ClearIsSelectedByKey();
}
else
{
ListView? dgv = menu.GetDataGridView();
if (dgv != null)
{
if (dgv.Items.Count > 0)
{
Select(dgv, (ListViewItemData)dgv.Items[0], true);
}
}
}
}
internal void ClearIsSelectedByKey()
{
ClearIsSelectedByKey(focussedMenu, focussedRow);
}
internal void Select(ListView dgv, ListViewItemData itemData, bool refreshview)
{
Menu menu = (Menu)dgv.GetParentWindow();
if (itemData != focussedRow || menu != focussedMenu)
{
ClearIsSelectedByKey();
}
focussedMenu = menu;
focussedRow = itemData;
itemData.data.IsSelected = true;
if (refreshview)
{
if (dgv.SelectedItems.Contains(itemData))
{
dgv.SelectedItems.Remove(itemData);
}
dgv.SelectedItems.Add(itemData);
}
}
private static void ClearIsSelectedByKey(Menu? menu, ListViewItemData? itemData)
{
if (menu != null && itemData != null)
{
ListView? dgv = menu?.GetDataGridView();
if (dgv != null)
{
if (dgv.SelectedItems.Contains(itemData))
{
dgv.SelectedItems.Remove(itemData);
}
itemData.data.IsSelected = false;
itemData.data.IsClicking = false;
}
}
}
private bool IsAnyMenuSelectedByKey(
ref Menu? subMenu,
ref string textSelected)
{
bool isStillSelected = false;
if (focussedRow != null)
{
ListViewItemData itemData = focussedRow;
RowData rowData = itemData.data;
if (rowData.IsSelected)
{
isStillSelected = true;
subMenu = rowData.SubMenu;
textSelected = itemData.ColumnText;
}
}
return isStillSelected;
}
2022-11-29 08:27:52 +13:00
private void SelectByKey(Key key, ModifierKeys modifiers, string keyInput = "", bool keepSelection = false)
{
int iRowBefore = focussedMenu?.GetDataGridView()?.Items.IndexOf(focussedRow) ?? -1;
Menu? menuBefore = focussedMenu;
ListViewItemData? rowBefore = focussedRow;
Menu? menu;
ListView? dgv;
ListView? dgvBefore;
2022-12-06 09:46:53 +13:00
Menu? menuFromSelected = null;
string textselected = string.Empty;
bool isStillSelected = IsAnyMenuSelectedByKey(ref menuFromSelected, ref textselected);
if (isStillSelected)
{
if (keepSelection)
{
// If current selection is still valid for this search then skip selecting different item
if (textselected.StartsWith(keyInput, true, CultureInfo.InvariantCulture))
{
return;
}
}
menu = focussedMenu;
dgv = menu?.GetDataGridView();
}
else
{
ResetSelectedByKey();
menu = null;
dgv = null;
}
dgvBefore = dgv;
bool toClear = false;
2022-11-29 08:27:52 +13:00
bool handled = false;
switch (key)
{
case Key.Enter:
if ((modifiers == ModifierKeys.None) && focussedRow != null && dgv != null)
{
ListViewItemData itemData = focussedRow;
RowData trigger = itemData.data;
if (trigger.IsMenuOpen || !trigger.IsPointingToFolder)
{
2023-04-25 08:38:36 +12:00
trigger.OpenItem(out bool doCloseAfterOpen);
if (doCloseAfterOpen)
{
ClosePressed?.Invoke();
}
}
else
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
EnterPressed?.Invoke(dgv, itemData);
}
2022-11-29 08:27:52 +13:00
handled = true;
}
break;
case Key.Up:
2022-11-29 08:27:52 +13:00
if ((modifiers == ModifierKeys.None) &&
2023-04-17 09:27:27 +12:00
dgv != null &&
(SelectMatchedReverse(dgv, focussedRow) ||
2022-11-29 08:27:52 +13:00
SelectMatchedReverse(dgv, dgv.Items.Count - 1)))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
2022-11-29 08:27:52 +13:00
handled = true;
}
break;
case Key.Down:
2022-11-29 08:27:52 +13:00
if ((modifiers == ModifierKeys.None) &&
(SelectMatched(dgv, focussedRow) ||
2022-11-29 08:27:52 +13:00
SelectMatched(dgv, 0)))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
2022-11-29 08:27:52 +13:00
handled = true;
}
break;
case Key.Home:
2022-11-29 08:27:52 +13:00
if ((modifiers == ModifierKeys.None) && SelectMatched(dgv, 0))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
2022-11-29 08:27:52 +13:00
handled = true;
}
break;
case Key.End:
2023-04-17 09:27:27 +12:00
if ((modifiers == ModifierKeys.None) &&
dgv != null &&
SelectMatchedReverse(dgv, dgv.Items.Count - 1))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
2022-11-29 08:27:52 +13:00
handled = true;
}
break;
case Key.Left:
2023-04-17 09:27:27 +12:00
if (modifiers == ModifierKeys.None &&
dgv != null &&
dgvBefore != null)
{
Menu? nextMenu = focussedMenu?.SubMenu;
2023-04-17 09:27:27 +12:00
bool nextMenuLocationIsLeft = nextMenu != null && menu != null && nextMenu.Location.X < menu.Location.X;
Menu? previousMenu = focussedMenu?.ParentMenu;
bool previousMenuLocationIsRight = previousMenu != null && menu != null && menu.Location.X < previousMenu.Location.X;
2022-11-29 08:27:52 +13:00
if (nextMenuLocationIsLeft || previousMenuLocationIsRight)
{
SelectNextMenu(iRowBefore, ref dgv, dgvBefore, menuFromSelected, isStillSelected, ref toClear);
}
else if (focussedMenu?.Level > 0)
2022-11-29 08:27:52 +13:00
{
SelectPreviousMenu(iRowBefore, ref menu, ref dgv, dgvBefore, ref toClear);
}
handled = true;
}
break;
case Key.Right:
2023-04-17 09:27:27 +12:00
if (modifiers == ModifierKeys.None &&
dgv != null &&
dgvBefore != null)
{
bool nextMenuLocationIsRight = focussedMenu?.SubMenu?.Location.X > focussedMenu?.Location.X;
bool previousMenuLocationIsLeft = focussedMenu?.Location.X > focussedMenu?.ParentMenu?.Location.X;
2022-11-29 08:27:52 +13:00
if (nextMenuLocationIsRight || previousMenuLocationIsLeft)
{
SelectNextMenu(iRowBefore, ref dgv, dgvBefore, menuFromSelected, isStillSelected, ref toClear);
}
else if (focussedMenu?.Level > 0)
2022-11-29 08:27:52 +13:00
{
SelectPreviousMenu(iRowBefore, ref menu, ref dgv, dgvBefore, ref toClear);
}
handled = true;
}
break;
case Key.Escape:
2022-11-29 08:27:52 +13:00
case Key.F4:
if ((key == Key.Escape && modifiers == ModifierKeys.None) ||
(key == Key.F4 && modifiers == ModifierKeys.Alt))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
ResetSelectedByKey();
2022-11-29 08:27:52 +13:00
toClear = true;
ClosePressed?.Invoke();
handled = true;
}
break;
default:
2022-11-29 08:27:52 +13:00
break;
}
if (!handled)
{
if (!string.IsNullOrEmpty(keyInput))
{
if (SelectMatched(dgv, focussedRow, keyInput) ||
2022-11-29 08:27:52 +13:00
SelectMatched(dgv, 0, keyInput))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, null);
SelectRow(dgv, focussedRow);
2022-11-29 08:27:52 +13:00
toClear = true;
}
else if (isStillSelected)
{
int prevRowIndex = focussedRow == null ? -1 : menuBefore?.GetDataGridView()?.Items.IndexOf(focussedRow) - 1 ?? -1;
focussedRow = prevRowIndex > 0 && menuBefore?.GetDataGridView()?.Items.Count > prevRowIndex ? (ListViewItemData?)menuBefore?.GetDataGridView()?.Items[prevRowIndex] : null;
if (SelectMatched(dgv, focussedRow, keyInput) ||
SelectMatched(dgv, 0, keyInput))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, null);
SelectRow(dgv, focussedRow);
}
2022-11-29 08:27:52 +13:00
else
{
focussedRow = rowBefore;
}
}
2022-11-29 08:27:52 +13:00
}
}
if (isStillSelected && toClear)
{
ClearIsSelectedByKey(menuBefore, rowBefore);
}
}
2023-04-17 09:27:27 +12:00
private void SelectPreviousMenu(int iRowBefore, ref Menu? menu, ref ListView? dgv, ListView? dgvBefore, ref bool toClear)
{
if (focussedMenu?.Level > 0)
{
if (focussedMenu.ParentMenu != null)
{
menu = focussedMenu = focussedMenu.ParentMenu;
focussedRow = null;
2023-04-17 09:27:27 +12:00
dgv = menu?.GetDataGridView();
if (dgv != null)
{
2023-04-17 09:27:27 +12:00
if (SelectMatched(dgv, dgv.Items.IndexOf(dgv.SelectedItems.Count > 0 ? dgv.SelectedItems[0] : null)) ||
SelectMatched(dgv, 0))
{
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
2023-04-17 09:27:27 +12:00
toClear = true;
}
}
}
}
else
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
ResetSelectedByKey();
toClear = true;
}
}
2023-04-17 09:27:27 +12:00
private void SelectNextMenu(int iRowBefore, ref ListView? dgv, ListView dgvBefore, Menu? menuFromSelected, bool isStillSelected, ref bool toClear)
{
if (isStillSelected)
{
if (menuFromSelected != null &&
menuFromSelected == focussedMenu?.SubMenu)
{
2023-04-17 09:27:27 +12:00
dgv = menuFromSelected?.GetDataGridView();
if (dgv != null && dgv.Items.Count > 0)
{
focussedMenu = menuFromSelected;
focussedRow = null;
if (SelectMatched(dgv, focussedRow) ||
SelectMatched(dgv, 0))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
}
}
}
}
else
{
focussedMenu = menus[0];
while (focussedMenu?.SubMenu != null)
{
focussedMenu = focussedMenu.SubMenu;
}
focussedRow = null;
Menu? lastMenu = focussedMenu;
2023-04-17 09:27:27 +12:00
if (lastMenu != null)
{
2023-04-17 09:27:27 +12:00
dgv = lastMenu?.GetDataGridView();
if (SelectMatched(dgv, focussedRow) ||
SelectMatched(dgv, 0))
{
2023-04-17 09:27:27 +12:00
RowDeselected?.Invoke(iRowBefore, dgvBefore);
SelectRow(dgv, focussedRow);
toClear = true;
}
}
}
}
private void SelectRow(ListView? dgv, ListViewItemData? itemData)
{
if (dgv != null && itemData != null)
2023-04-17 09:27:27 +12:00
{
InUse = true;
RowSelected?.Invoke(dgv, itemData);
2023-04-17 09:27:27 +12:00
}
}
private bool SelectMatched(ListView? dgv, ListViewItemData? start, string keyInput = "") =>
start != null && dgv != null && SelectMatched(dgv, dgv.Items.IndexOf(start), keyInput);
2023-04-17 09:27:27 +12:00
private bool SelectMatched(ListView? dgv, int indexStart, string keyInput = "")
{
bool found = false;
if (dgv != null && indexStart >= 0)
{
for (uint i = (uint)indexStart; i < dgv.Items.Count; i++)
{
2023-04-17 09:27:27 +12:00
if (Select(dgv, i, keyInput))
{
found = true;
break;
}
}
}
return found;
}
private bool SelectMatchedReverse(ListView dgv, ListViewItemData? start, string keyInput = "") =>
start != null && SelectMatchedReverse(dgv, dgv.Items.IndexOf(start), keyInput);
private bool SelectMatchedReverse(ListView dgv, int indexStart, string keyInput = "")
{
bool found = false;
if (indexStart > 0)
{
for (int i = indexStart; i > -1; i--)
{
if (Select(dgv, (uint)i, keyInput))
{
found = true;
break;
}
}
}
return found;
}
private bool Select(ListView dgv, uint i, string keyInput = "")
{
bool found = false;
if (dgv.Items.Count > i && dgv.Items[(int)i] != focussedRow)
{
ListViewItemData itemData = (ListViewItemData)dgv.Items[(int)i];
2022-11-26 11:31:20 +13:00
if (itemData.ColumnText.StartsWith(keyInput, true, CultureInfo.InvariantCulture))
{
focussedRow = itemData;
itemData.data.IsSelected = true;
2022-11-26 11:31:20 +13:00
if (dgv.SelectedItems.Contains(itemData))
{
2022-11-26 11:31:20 +13:00
dgv.SelectedItems.Remove(itemData);
}
2022-11-26 11:31:20 +13:00
dgv.SelectedItems.Add(itemData);
dgv.ScrollIntoView(itemData);
found = true;
}
}
return found;
}
}
}