我想创建一个应用程序,我在窗口上绘制,无论是窗口还是全屏,我抓住了鼠标,但没有拦截任何 WM键盘快捷键,如Alt + Tab,我还需要在用户进入/离开焦点时收到通知。
Google Chrome,Firefox或gnome-terminal等常见应用程序可以很好地处理这个问题(使用F11全屏,但仍然使用Alt + Tab),但他们不会抓住鼠标。
SDL对此用例的处理方式臭名昭着:SDL_WM_GrabInput抓取鼠标但也拦截了WM快捷方式;和SDL_FULLSCREEN似乎有一些自动抓取(不要问我为什么)。
解决方案可能是自己为Alt + Tab编写代码,但这很糟糕(并且对其他WM快捷方式没有帮助,例如更改为其他工作区)。
另一个解决方案是不调用SDL_WM_GrabInput,而是伪造一个抓取:只需隐藏鼠标指针(使用SDL_ShowCursor)并在用户移动时将其移回中心。这是丑陋的,但在实践中工作 - 当然除了SDL_FULLSCREEN,因为它自动抓取(不同于理智的实现)。这是一个全屏功能的SDL解决方案,但这仍然不是我想要的。我不想有黑客来启用和禁用抓取,我想抓住鼠标而不是抓住键盘。
所以我对SDL很生气,并希望看到其他选择。我希望使用SDL,但这不是必需的。
This question似乎指出SDL实际上做的是使用XGrabKeyboard。通过阅读手册页,我不清楚你是否可以在不抓取键盘的情况下抓住鼠标(我自己从未使用过Xlib)。
我知道如何使用GTK制作“假全屏”(即Alt + Tab友好,gnome-terminal类型)。我想这样做,加上鼠标隐藏并将其移回中心(“假抓取”)可以做到这一点,但这感觉就像是太多的胶带。必须有一个更简单的方法。 (另外我不想将GTK添加为依赖项;但我也不确定是否进行原始Xlib调用是个好主意。)
对此有什么好的解决方案?
我需要一个Linux / X11解决方案但是跨平台它会很好 - 我知道这可以在Windows上顺利解决,所以也许有一个库正是这样做的。 (另外,我使用OpenGL渲染,但这无关紧要)
PS:也许我对这个问题了解不多,而且我没有问正确的问题,所以请随意指出我没有考虑过的方法。答案 0 :(得分:4)
这些天我一直在使用Gtk和GtkGLExt在Linux上进行游戏(例如,我的Ludum Dare条目)。这里(gtk.c Gist)是我的代码,我已经在FreeBSD许可下发布了它。您可以看到我为调试窗口状态更改而注释掉的代码。我应该提一下,这是一个更大的框架的一部分,我用于Ludum Dare之类的东西,我更喜欢SDL,原因与你提到的相同:生成的应用程序更符合用户对本机应用程序的期望他们的平台(Linux,OS X,Windows)。毋庸置疑,走这条路是很多工作。
然而,原始X11编程真是一团糟。我估计我需要几周的时间来编写和阅读文档才能将我的Gtk代码转换为X11代码;这对我来说不值得,但你可能会有不同的决定。 (很多时候只是为了消除每个人都安装的依赖!)Gtk依赖实际上并不是那么离谱,它可能已经加载到内存中了,而且2.x的ABI非常稳定所以它不会损害二进制兼容性X11游戏编程的一个大问题是事件处理是一团糟。您无法轻松轮询事件。 Xlib接口是阻塞的,因此您必须使用一个神秘的函数调用序列来读取数据,检查是否存在待处理事件,然后只读取队列中存在的事件(否则您的应用程序将阻塞直到事件出现)。 xcb库是Xlib的替代品,它更好,并支持非阻塞接口,但它不能很好地与OpenGL一起使用。所以你可以尝试混合两者,或者你可以使用Gtk。
让我给你一个全屏Gtk代码片段:
static void toggle_fullscreen()
{
if (sg_gtk_status & SG_STATUS_VISIBLE) {
if (sg_gtk_status & SG_STATUS_FULLSCREEN)
gtk_window_unfullscreen(sg_window);
else
gtk_window_fullscreen(sg_window);
}
}
想象一下X11界面会有多少工作:弄清楚屏幕的大小(可能有不止一个!),调整窗口大小,改变排序,使其在其他一切之上,改变装饰,然后在用户切换虚拟桌面时智能响应。胡说! (实际上,这描述了我的代码如何在OS X上运行,但API更好。)
更重要的是,如果您使用的是X11,则必须学习如何以预期的方式对用户做出反应。 X11是从20世纪80年代开始的。相比之下,Gtk为您的应用程序提供了大量适当的默认设置,来自Gtk的UI设计师。 UI设计是有效的。请记住:你可以将X11和Gtk混合使用。
摘要: X11编程适用于有大量业余时间的程序员。
注意: GtkGLExt添加了一个恼人的链接器标志-Wl,--export-dynamic
。我的构建脚本从pkg-config
的输出中删除了该标志。
X11体验我认为我花了大约一个星期试图让整个事情在X11中运行,最后我在这个过程中遇到了很多死胡同。从我的失败中学习。