Magpie/src/App/ShortcutControl.cpp

250 lines
7.1 KiB
C++

#include "pch.h"
#include "ShortcutControl.h"
#if __has_include("ShortcutControl.g.cpp")
#include "ShortcutControl.g.cpp"
#endif
using namespace winrt;
using namespace Windows::UI::Xaml::Controls;
using namespace Windows::UI::Xaml::Media;
namespace winrt::Magpie::App::implementation {
const DependencyProperty ShortcutControl::ActionProperty = DependencyProperty::Register(
L"Action",
xaml_typename<HotkeyAction>(),
xaml_typename<Magpie::App::ShortcutControl>(),
PropertyMetadata(box_value(HotkeyAction::COUNT_OR_NONE), &ShortcutControl::_OnActionChanged)
);
const DependencyProperty ShortcutControl::_IsErrorProperty = DependencyProperty::Register(
L"_IsError",
xaml_typename<bool>(),
xaml_typename<Magpie::App::ShortcutControl>(),
PropertyMetadata(box_value(false), nullptr)
);
ShortcutControl* ShortcutControl::_that = nullptr;
ShortcutControl::ShortcutControl() {
InitializeComponent();
App app = Application::Current().as<App>();
_settings = app.Settings();
_hotkeyManager = app.HotkeyManager();
_hotkeyChangedRevoker = _settings.HotkeyChanged(
auto_revoke, { this,&ShortcutControl::_Settings_OnHotkeyChanged });
_shortcutDialog.Title(box_value(L"激活快捷键"));
_shortcutDialog.Content(_shortcutDialogContent);
_shortcutDialog.PrimaryButtonText(L"保存");
_shortcutDialog.CloseButtonText(L"取消");
_shortcutDialog.DefaultButton(ContentDialogButton::Primary);
_shortcutDialog.Opened({ this, &ShortcutControl::ShortcutDialog_Opened });
_shortcutDialog.Closing({ this, &ShortcutControl::ShortcutDialog_Closing });
}
IAsyncAction ShortcutControl::EditButton_Click(IInspectable const&, RoutedEventArgs const&) {
_previewHotkey.CopyFrom(_hotkey);
_shortcutDialogContent.Keys(_previewHotkey.GetKeyList());
_shortcutDialog.XamlRoot(XamlRoot());
_shortcutDialog.RequestedTheme(ActualTheme());
// 防止快速点击时崩溃
static bool showing = false;
if (showing) {
co_return;
}
showing = true;
co_await _shortcutDialog.ShowAsync();
showing = false;
}
void ShortcutControl::ShortcutDialog_Opened(ContentDialog const&, ContentDialogOpenedEventArgs const&) {
_that = this;
_keyboardHook = SetWindowsHookEx(WH_KEYBOARD_LL, _LowLevelKeyboardProc, NULL, 0);
_previewHotkey.CopyFrom(_hotkey);
_shortcutDialogContent.Keys(_previewHotkey.GetKeyList());
_shortcutDialogContent.IsError(IsError());
_shortcutDialog.IsPrimaryButtonEnabled(!IsError());
_pressedKeys.Clear();
}
void ShortcutControl::ShortcutDialog_Closing(ContentDialog const&, ContentDialogClosingEventArgs const& args) {
UnhookWindowsHookEx(_keyboardHook);
if (args.Result() == ContentDialogResult::Primary) {
_settings.SetHotkey(Action(), _previewHotkey);
}
}
HotkeyAction ShortcutControl::Action() const {
return GetValue(ActionProperty).as<HotkeyAction>();
}
void ShortcutControl::Action(HotkeyAction value) {
SetValue(ActionProperty, box_value(value));
}
void ShortcutControl::_IsError(bool value) {
SetValue(_IsErrorProperty, box_value(value));
}
bool ShortcutControl::IsError() const {
return GetValue(_IsErrorProperty).as<bool>();
}
bool CheckVirtualKey(DWORD vkCode) {
return (vkCode >= 'A' && vkCode <= 'Z') // 字母
|| (vkCode >= '0' && vkCode <= '9') // 数字(顶部)
|| (vkCode >= VK_NUMPAD0 && vkCode <= VK_NUMPAD9) // 数字(小键盘)
|| (vkCode >= VK_F1 && vkCode <= VK_F24) // F1~F24
|| (vkCode >= VK_SPACE && vkCode <= VK_DOWN) // 空格、Page Up/Down、End、Home、方向键
|| vkCode == VK_INSERT // Insert
|| vkCode == VK_DELETE // Delete
|| vkCode == VK_ADD // 加(小键盘)
|| vkCode == VK_SUBTRACT // 减(小键盘)
|| vkCode == VK_MULTIPLY // 乘(小键盘)
|| vkCode == VK_DIVIDE // 除(小键盘)
|| (vkCode >= VK_OEM_1 && vkCode <= VK_OEM_3) // 分号、等号、逗号、-、句号、/、`
|| (vkCode >= VK_OEM_4 && vkCode <= VK_OEM_7) // [、\、]、'
|| vkCode == VK_BACK // Backspace
|| vkCode == VK_RETURN; // 回车
}
event_token ShortcutControl::PropertyChanged(Data::PropertyChangedEventHandler const& value) {
return _propertyChangedEvent.add(value);
}
void ShortcutControl::PropertyChanged(event_token const& token) {
_propertyChangedEvent.remove(token);
}
LRESULT ShortcutControl::_LowLevelKeyboardProc(int nCode, WPARAM wParam, LPARAM lParam) {
if (nCode != HC_ACTION || !_that) {
return CallNextHookEx(NULL, nCode, wParam, lParam);
}
// 只有位于前台时才监听按键
App app = Application::Current().as<App>();
if (GetForegroundWindow() != (HWND)app.HwndHost()) {
return CallNextHookEx(NULL, nCode, wParam, lParam);
}
bool isKeyDown = wParam == WM_KEYDOWN || wParam == WM_SYSKEYDOWN;
DWORD code = ((KBDLLHOOKSTRUCT*)lParam)->vkCode;
switch (code) {
case VK_LWIN:
case VK_RWIN:
_that->_pressedKeys.Win(isKeyDown);
break;
case VK_CONTROL:
case VK_LCONTROL:
case VK_RCONTROL:
_that->_pressedKeys.Ctrl(isKeyDown);
break;
case VK_SHIFT:
case VK_LSHIFT:
case VK_RSHIFT:
_that->_pressedKeys.Shift(isKeyDown);
break;
case VK_MENU:
case VK_LMENU:
case VK_RMENU:
_that->_pressedKeys.Alt(isKeyDown);
break;
default:
{
if (CheckVirtualKey(code)) {
if (isKeyDown) {
_that->_pressedKeys.Code(code);
} else {
_that->_pressedKeys.Code(0);
}
} else {
// 不处理的键位
isKeyDown = false;
}
break;
}
}
if (isKeyDown) {
Magpie::App::HotkeySettings& previewHotkey = _that->_previewHotkey;
previewHotkey.CopyFrom(_that->_pressedKeys);
_that->_shortcutDialogContent.Keys(previewHotkey.GetKeyList());
bool isError = false;
bool isPrimaryButtonEnabled = false;
if (previewHotkey.Equals(_that->_hotkey) && !_that->IsError()) {
isError = false;
isPrimaryButtonEnabled = true;
} else {
UINT modCount = 0;
if (previewHotkey.Code() == 0) {
if (previewHotkey.Win()) {
++modCount;
}
if (previewHotkey.Alt()) {
++modCount;
}
if (modCount <= 1 && previewHotkey.Ctrl()) {
++modCount;
}
if (modCount <= 1 && previewHotkey.Shift()) {
++modCount;
}
}
if (modCount == 1) {
// Modifiers 个数为 1 时不显示错误
isError = false;
isPrimaryButtonEnabled = false;
} else {
isError = !previewHotkey.Check();
isPrimaryButtonEnabled = !isError;
}
}
_that->_shortcutDialogContent.IsError(isError);
_that->_shortcutDialog.IsPrimaryButtonEnabled(isPrimaryButtonEnabled);
}
return -1;
}
void ShortcutControl::_OnActionChanged(DependencyObject const& sender, DependencyPropertyChangedEventArgs const&) {
ShortcutControl* that = get_self<ShortcutControl>(sender.as<default_interface<ShortcutControl>>());
that->_UpdateHotkey();
that->_propertyChangedEvent(*that, PropertyChangedEventArgs{ L"Action" });
}
void ShortcutControl::_Settings_OnHotkeyChanged(IInspectable const&, HotkeyAction action) {
if (action == Action()) {
_UpdateHotkey();
}
}
void ShortcutControl::_UpdateHotkey() {
HotkeyAction action = Action();
HotkeySettings hotkey = _settings.GetHotkey(action);
if (hotkey) {
_hotkey.CopyFrom(hotkey);
// 此时 HotkeyManager 中的回调已执行
_IsError(_hotkeyManager.IsError(action));
} else {
_hotkey.Clear();
_IsError(false);
}
KeysControl().ItemsSource(_hotkey.GetKeyList());
}
}