- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我想制作一个模拟按键的程序。我想我基本上已经完成了,但我猜我做错了什么,因为它没有按照我的预期去做。我做了一个小示例程序来说明这个问题。主要问题是,如果我想生成大写字母,它不适用于像 'zZ'
这样的字符串。它只生成小写字母 'zz'
。尽管像 '! 这样的符号$ & _ >'
等工作正常(这需要在我的德语键盘布局上进行 shift),甚至像 '💣' 这样的多字节。我正在做的是:
序言:
所以基本上,模拟按键的主要问题首先是布局因用户而异,最重要的是修改键。因此,如果您走天真的路线并使用 XStringToKeysym()
获取键码,则使用 XKeysymToKeycode()
从该键码获取键码并触发该事件,它不像大多数“新手”那样工作' 会期望(像我一样)。这里的问题是,多个键符被映射到相同的键码。就像 'a'
和 'A'
的 keysysm 被映射到相同的键码,因为它们位于键盘上链接到该键码的相同物理按钮上。因此,如果你从上面走这条路,你最终会得到相同的键码,尽管键符不同但映射到相同的按钮/键码。通常没有办法解决这个问题,因为不清楚 'A'
最初是如何存在的。 shift+a 或 caps+a 或者你有一个漂亮的键盘,上面有一个 'a'
和 'A'
按钮。另一个问题是我如何为运行该应用程序的人的键盘上什至不存在的按钮发出按键。如果我想输入 'Ä'
(德语变音符号),就像在英语布局上按下什么键一样。这不起作用,因为 XKeysymToKeycode()
不会为此返回正确的键码,因为在该布局中没有键符映射。
我的方法:
为了避免这种情况,我要做的是找到一个未被使用的键码。您可以使用 255-8 个键码,但普通键盘上只有约 110 个键,因此通常还剩下一些空间。我试图找到那些未映射到当前布局的键码之一,并用它来分配我自己的键符。然后我从我的 char 中得到一个 keysym,我通过遍历我的字符串并将它传递给 XStringToKeysym()
这给了我适当的 keysym。在大多数情况下,'💣' 未映射到我所知道的任何键盘布局。所以我将它映射到未使用的键码并使用 XTestFakeKeyEvent()
按下它并为字符串中的每个字符重复该操作。这适用于人们能想到的所有花哨的字形,但它不适用于简单的字母,我真的不知道为什么 :( 在我的调试 session 中,keysyms 和 keycodes 似乎是正确的,只是 XTestFakeKeyEvent()
在那种情况下没有做正确的事情。可能是我在键盘映射部分搞砸了,但我不太确定这里的问题是什么,我希望有人有好主意,可以帮助我找到解决方法一个可行的解决方案。
我只是在 strings
数组中使用这个 unicode 表示法,因为我不想在此处的示例中处理它。假设有代码从任意输入字符串生成它。
请注意,下面的代码可能会破坏您的键盘映射,使您无法再键入和使用键盘,需要重新启动您的 X-Server/PC ...我希望它能不在当前状态(在这里工作正常),如果您摆弄代码请注意
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <X11/X.h>
#include <X11/Xlib.h>
#include <X11/extensions/XTest.h>
#include <unistd.h>
//gcc -g enigo2.c -lXtst -lX11
int main(int argc, char *argv[])
{
Display *dpy;
dpy = XOpenDisplay(NULL);
//my test string already transformed into unicode
//ready to be consumed by XStringToKeysym
const char *strings[] = {
"U1f4a3",// 💣
"U007A", //z
"U005A", //Z
"U002f", //'/'
"U005D", //]
"U003a", //:
"U002a", //*
"U0020", //' '
"U0079", //y
"U0059", //Y
"U0020", //' '
"U0031", //1
"U0021", //!
"U0020", //' '
"U0036", //6
"U0026", //&
"U0020", //' '
"U0034", //4
"U0024", //$
"U0020", //' '
"U002D", //-
"U005F", //_
"U0020", //' '
"U003C", //<
"U003E", //>
"U0063", //c
"U0043", //C
"U006f", //o
"U004f", //O
"U00e4", //ä
"U00c4", //Ä
"U00fc", //ü
"U00dc", //Ü
};
KeySym *keysyms = NULL;
int keysyms_per_keycode = 0;
int scratch_keycode = 0; // Scratch space for temporary keycode bindings
int keycode_low, keycode_high;
//get the range of keycodes usually from 8 - 255
XDisplayKeycodes(dpy, &keycode_low, &keycode_high);
//get all the mapped keysyms available
keysyms = XGetKeyboardMapping(
dpy,
keycode_low,
keycode_high - keycode_low,
&keysyms_per_keycode);
//find unused keycode for unmapped keysyms so we can
//hook up our own keycode and map every keysym on it
//so we just need to 'click' our once unmapped keycode
int i;
for (i = keycode_low; i <= keycode_high; i++)
{
int j = 0;
int key_is_empty = 1;
for (j = 0; j < keysyms_per_keycode; j++)
{
int symindex = (i - keycode_low) * keysyms_per_keycode + j;
// test for debugging to looking at those value
// KeySym sym_at_index = keysyms[symindex];
// char *symname;
// symname = XKeysymToString(keysyms[symindex]);
if(keysyms[symindex] != 0) {
key_is_empty = 0;
} else {
break;
}
}
if(key_is_empty) {
scratch_keycode = i;
break;
}
}
XFree(keysyms);
XFlush(dpy);
usleep(200 * 1000);
int arraysize = 33;
for (int i = 0; i < arraysize; i++)
{
//find the keysym for the given unicode char
//map that keysym to our previous unmapped keycode
//click that keycode/'button' with our keysym on it
KeySym sym = XStringToKeysym(strings[i]);
KeySym keysym_list[] = { sym };
XChangeKeyboardMapping(dpy, scratch_keycode, 1, keysym_list, 1);
KeyCode code = scratch_keycode;
usleep(90 * 1000);
XTestFakeKeyEvent(dpy, code, True, 0);
XFlush(dpy);
usleep(90 * 1000);
XTestFakeKeyEvent(dpy, code, False, 0);
XFlush(dpy);
}
//revert scratch keycode
{
KeySym keysym_list[] = { 0 };
XChangeKeyboardMapping(dpy, scratch_keycode, 1, keysym_list, 1);
}
usleep(100 * 1000);
XCloseDisplay(dpy);
return 0;
}
最佳答案
当您将给定键码的单个键符发送到 XChangeKeyboardMapping
并且它是一个字母时,它会自动为 shift 和 capslock 修饰符填充正确的大写和小写等价物。也就是说,经过
XChangeKeyboardMapping(dpy, scratch_keycode, 1, &keysym, 1);
scratch_keycode 的键码映射(在我的机器上)有效地更改为
tolower(keysym), toupper(keysym), tolower(keysym), toupper(keysym), tolower(keysym), toupper(keysym), 0, 0, 0, 0, ...
为了抑制这种行为,每个键码发送 2 个相同的键符:
KeySym keysym_list[2] = { sym, sym };
XChangeKeyboardMapping(dpy, scratch_keycode, 2, keysym_list, 1);
这将使用相同的键符填充移位和未移位的位置。
关于C xtest 为每个 Unicode 字符发出按键,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44313966/
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求提供代码的问题必须表现出对所解决问题的最低限度理解。包括尝试过的解决方案、为什么它们不起作用,以及预
为什么在 C# 中添加两个 char 结果是 int 类型? 例如,当我这样做时: var pr = 'R' + 'G' + 'B' + 'Y' + 'P'; pr 变量变为 int 类型。我希望它是
下面的代码可以编译,但 char 类型的行为与 int 类型的行为不同。 特别是 cout ::ikIsX >() ::ikIsX >() ::ikIsX >() using names
我正在寻找一个正则表达式,它可以匹配长度为 1 个或多个字符但不匹配 500 的内容。这将在 Rails 路由文件中使用,特别是用于处理异常。 路线.rb match '/500', to: 'err
对于 C 编程作业,我正在尝试编写几个头文件来检查所谓的“X 编程语言”的语法。我最近才开始,正在编写第一个头文件。这是我编写的代码: #ifndef _DeclarationsChecker_h_
为什么扩展的 ascii 字符(â、é 等)被替换为 字符? 我附上了一张图片...但我正在使用 PHP 从 MySQL 中提取数据,其中一些位置有扩展字符...我使用的是 Arial 字体。 您可以
我有一个与 R 中的断线相关的简单问题。 我正在尝试粘贴,但在获取(字符/数字)之间的断线时遇到问题。请注意,这些值包含在向量中(V1=81,V2=55,V3=25)我已经尝试过这段代码: cat(p
如何将 ANSI 字符 (char) 转换为 Unicode 字符 (wchar_t),反之亦然? 是否有用于此目的的任何跨平台源代码? 最佳答案 是的,在 中你有mbstowcs()和 wcsto
函数 fromCharCode 不适用于国际 ANSI 字符。例如,对于 ID 为 192 到 223 的俄语 ANSI (cp-1251) 字符,它返回特殊字符。如何解决这个问题? 我认为,需要将A
如果不喜欢,我想隐藏 id,但不起作用 SELECT * FROM character, character_actor WHERE character.id NOT LIKE character_a
现在这个程序成功地反转了键盘输入的单词。但是我想在我反转它之前“保存”指针中的单词,所以我可以比较两者,反转的和“原始的”,并检查它们是否是回文。我还没有太多经验,可能会出现比我知道的更多的错误,但我
Memcpy 和 memcmp 函数可以接受指针变量吗? char *p; char* q; memcpy(p,q,10); //will this work? memcmp(p,q,10); //w
恐怕我对一个相当过饱和的主题的细节有疑问,我搜索了很多,但找不到一个明确的答案来解决这个特定的明显-imho-重要的问题: 使用UTF-8将byte[]转换为String时,每个字节(8bit)都变成
我有一个奇怪的问题。我需要从 stat 命令打印输出字符串。 我已经编写了获取一些信息的代码。 import glob import os for file in glob.glob('system1
我正在使用 Java 并具有其值如下所示的字符串, String data = "vale-cx"; data = data.replaceAll("\\-", "\\-\\"); 我正在替换其中的“
String urlParameters = "login=test&password=te&ff"; 我有一个String urlParams,& - 是密码的一部分,如何使其转义,从而不被识别为分
大家好,我只想从此字符串中提取第一个字母: String str = "使 徒 行 傳 16:31 ERV-ZH"; 我只想获取这些字符: 使 徒 行 傳 并且不包括 ERV-ZH 仅数
这个问题已经有答案了: Crash or "segmentation fault" when data is copied/scanned/read to an uninitialized point
所以, 我有一个字符**;它本质上是一个句子,带有指向该句子中每个单词的指针;即 'h''i''\0''w''o''r''l''d''\0''y''a''y''!''\0' 在这种情况下,我希望使用可
这个问题在这里已经有了答案: Using quotation marks inside quotation marks (12 个答案) 关闭 7 年前。 如何打印 " 字符? 我知道打印 % 符号
我是一名优秀的程序员,十分优秀!