7

我想创建一个应用程序,我在窗口上绘制,无论是窗口还是全屏,我抓住鼠标但拦截任何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,但这不是必需的。

这个问题似乎在指出,SDL 实际上所做的是使用 XGrabKeyboard。通过阅读手册页,我并不清楚您是否可以在不抓住键盘的情况下抓住鼠标(我自己从未使用过 Xlib)。

我知道如何使用 GTK(即 Alt+Tab 友好的 gnome-terminal 品种)制作“假全屏”。我想这样做,再加上鼠标隐藏并将其移回中心(“假抓取”)可以做到这一点,但这感觉就像太多的胶带。必须有更简单的方法。(此外,我不想将 GTK 添加为依赖项;但我也不确定进行原始 Xlib 调用是否是个好主意)。

对此有什么好的解决方案?

我需要一个 Linux/X11 解决方案,但它最好是跨平台的 - 我知道这可以在 Windows 上顺利解决,所以也许有一个库可以做到这一点。(另外,我用 OpenGL 渲染,但这无关紧要)

PS:也许我对这个问题的理解很差,我没有问正确的问题,所以请随时指出我没有考虑过的方法。

4

1 回答 1

5

这些天来,我一直在使用 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 来自 1980 年代。相比之下,Gtk 为您的应用程序提供了来自 Gtk 的 UI 设计人员的一系列适当的默认值。UI设计是工作。请记住:您可以将 X11 与 Gtk 混合使用。

总结: X11 编程是为有大量空闲时间的程序员准备的。

Notes: The GtkGLExt adds an annoying linker flag, -Wl,--export-dynamic. My build scripts remove the flag from pkg-config's output.

X11 experiences: I think I spent about a week trying to get the whole thing working in X11, and I ended up following lots of dead ends in the process. Learn from my failures.

于 2013-02-13T04:57:14.990 回答