C# Registerhotkey() 不会读取某些键码



在尝试使用Visual Studio Windows表单创建自动点击器应用程序以允许您选择所需的热键时,我偶然发现了一个问题

当用户输入诸如~/之类的键时,热键将不起作用,但是,如果我输入0xC0(对于波浪号)或0xDC(用于斜杠)而不是(int)[波浪号]或(int)'/'它将起作用。

最重要的是字母必须大写,但这没关系,因为我可以使用 ToUpper() 将其大写

允许用户选择字符的部分很好,它将字母作为字符串返回

在这里,我实现了RegisterHotKey()和UnregisterHotKey()以及为表单设置事件:

[DllImport("user32.dll")]
public static extern bool RegisterHotKey(IntPtr hWnd, int id, int fsModifiers, int vlc);
[DllImport("user32.dll")]
public static extern bool UnregisterHotKey(IntPtr hWnd, int id);
protected override void WndProc(ref Message m) {
if (m.Msg == 0x0312 && m.WParam.ToInt32() == 1)
{
toggleClicker(); //This turns my autoclicker off and on
}
base.WndProc(ref m);
}
public Form1(){
InitializeComponent();
/* Ignore this:
VersionText.Text = "Version " + version;
comboBox1.SelectedIndex = 0;
*/
RegisterHotKey(this.Handle, 1, 0x0002, (int)Keys.P); //This is the default and it works just fine
}
private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
UnregisterHotKey(this.Handle, 1);
}

这是用户选择他们想要的密钥时的代码

private void button2_Click(object sender, EventArgs e)
{
/*Ignore this
if (isClickerRunning)
{
//Console.WriteLine("Stopping...");
clickerThread.Abort();
isClickerRunning = false;
}
*/
int Prefix = 0;
//Set Prefix to the value chosen in the combobox
if (comboBox1.SelectedIndex == 1) { Prefix = 0x0002; } //Ctrl
if (comboBox1.SelectedIndex == 2) { Prefix = 0x0004; } //Shift
if (comboBox1.SelectedIndex == 3) { Prefix = 0x0001; } //Alt
if (comboBox1.SelectedIndex == 4) { Prefix = 0x0002 + 0x0004; } //Ctrl + Shift
if (comboBox1.SelectedIndex == 5) { Prefix = 0x0002 + 0x0001; } //Ctrl + Alt
if (comboBox1.SelectedIndex == 6) { Prefix = 0x0002 + 0x0001 + 0x0004; } //Ctrl + Alt + Shift

bool Success1 = UnregisterHotKey(this.Handle, 1);
bool Success2 = RegisterHotKey(this.Handle, 1, prefix, (int)KeyCharacter); //KeyCharacter is the key that the user has selected
//Test if RegisterHotKey and UnregisterHotKey failed
if(Success1 == false || Success2 == false)
{
//Set the hotkey back to default if it failed
Success1 = UnregisterHotKey(this.Handle, 1);
Success2 = RegisterHotKey(this.Handle, 1, 0x0002, (int)Keys.P);
}
//Test if it failed again
if(Success1 == false || Success2 == false)
{
MessageBox.Show("FATAL ERROR!nCould not register hotkey. Quiting...", "FATAL ERROR", MessageBoxButtons.OK, MessageBoxIcon.Error);
Application.Exit();
}
}

程序启动时,我有一个默认的热键CTL + P设置 默认热键工作正常,但是当我在热键选择文本框中输入类似/的键并通过单击按钮进行设置时,它将阻止默认热键工作,但是当我按下热键时,我只是设置它不起作用。但是它适用于大写的A,例如

基本上我要问的是我怎样才能把~变成0xC0或/变成0xDC

任何帮助不胜感激

您需要使用VkKeyScanWin32 API 函数来获取字符的虚拟键代码。

那么,在您的情况下,您将在调用RegisterHotKey之前用您的KeyCharacter调用下面的VirtualKeyCodeFromChar函数。

实际上最好使用VkKeyScanEx功能,因为键盘布局不同,因此 OEM 键可能会因为布局不同而为同一字符生成不同的键代码。但我会让你自己探索使用该函数。同时。。。

using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
namespace ConsoleApp1
{
class Program
{
[DllImport("user32.dll")] static extern short VkKeyScan(char c);
static void Main(string[] args)
{
var s = "~|-.";
foreach(var c in s)
{
var key = VirtualKeyCodeFromChar(c);
Console.WriteLine($"{c}, {key}, 0x{key:X}n");
}
Console.ReadKey();
}
static int VirtualKeyCodeFromChar(char c)
{
var composite = VkKeyScan(c);
byte keyCode = (byte)(composite & 255);
byte shiftState = (byte)((composite >> 8) & 255);
Keys key = (Keys)keyCode;
if ((shiftState & 1) != 0) key |= Keys.Shift;
if ((shiftState & 2) != 0) key |= Keys.Control;
if ((shiftState & 4) != 0) key |= Keys.Alt;
return (int)key;
}
}
}

最新更新