10

我正在将一些代码从 Windows 移植到 XLib。在 windows 代码中,我可以通过调用InvalidateRect然后处理相应的WM_PAINT消息来强制重绘。但是,我无法在 X11/XLib 中找到如何执行此操作。我看到有一条Expose消息,但不确定是否是同一件事。

如果这很重要,我需要这样做以强制窗口以特定帧速率为基于 OpenGL 的程序呈现。

4

2 回答 2

5

为了稍微扩展 BЈовић 给出的有用答案,

使用原始 Xlib,您可以随时在单个线程中进行绘制,因为每个 Xlib 函数都指定了完整的显示、窗口和上下文。AFAIK,多线程所有赌注都没有了。

如果您在桌面环境中,您还必须有一个 Expose 事件处理程序,并选择这些事件。即使您正在编写一个全屏程序,拥有一个也不会受到伤害。

大多数工具包没有那么灵活,只能在指定的事件处理程序中绘制(但在许多其他方面使用起来更好),并且有一些等效于 Windows InvalidateRect。在原始 Xlib 中,您可以通过向自己发送 Expose 事件来获得相同的效果。这样做不会导致任何真正的性能问题,并且会使代码更容易被其他程序员理解,并且更容易移植,所以你也可以。

还有 XClearArea 和 XClearWindow 函数会为您生成 Expose 事件,但它们首先会使用背景颜色擦除部分/全部,这可能会导致闪烁。

使用 OpenGL 会变得有点复杂,因为您还必须使用 GLX。我在 http://cs.anu.edu.au/~hugh.fisher/3dteach/ 上有一个非常简单的 OpenGL/Xlib 程序,作为示例可能很有用。

于 2013-06-11T02:43:07.987 回答
4

您需要处理 Expose 事件。本教程通过示例说明如何处理 Expose 事件:

#include <stdio.h>
#include <stdlib.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xos.h>
#include <X11/Xatom.h>
#include <X11/keysym.h>
/*Linux users will need to add -ldl to the Makefile to compile 
 *this example.
 */
Display *dis;
Window win;
XEvent report;
GC green_gc;
XColor green_col;
Colormap colormap;
/*
Try changing the green[] = below to a different color.
The color can also be from /usr/X11R6/lib/X11/rgb.txt, such as RoyalBlue4.
A # (number sign) is only needed when using hexadecimal colors.
*/
char green[] = "#00FF00";

int main() {
    dis = XOpenDisplay(NULL);
    win = XCreateSimpleWindow(dis, RootWindow(dis, 0), 1, 1, 500, 500, 0, BlackPixel (dis, 0), BlackPixel(dis, 0));
    XMapWindow(dis, win);
    colormap = DefaultColormap(dis, 0);
    green_gc = XCreateGC(dis, win, 0, 0);
    XParseColor(dis, colormap, green, &green_col);
    XAllocColor(dis, colormap, &green_col);
    XSetForeground(dis, green_gc, green_col.pixel);

    XSelectInput(dis, win, ExposureMask | KeyPressMask | ButtonPressMask);

    XDrawRectangle(dis, win, green_gc, 1, 1, 497, 497);
    XDrawRectangle(dis, win, green_gc, 50, 50, 398, 398);
    XFlush(dis);

    while (1)  {
    XNextEvent(dis, &report);
        switch  (report.type) {
        case Expose:   
            fprintf(stdout, "I have been exposed.\n");
                XDrawRectangle(dis, win, green_gc, 1, 1, 497, 497);
                XDrawRectangle(dis, win, green_gc, 50, 50, 398, 398);
                XFlush(dis);
            break;
            case KeyPress:
        /*Close the program if q is pressed.*/
                if (XLookupKeysym(&report.xkey, 0) == XK_q) {
                exit(0);
                }
            break;
        }
    }

return 0;
}

我可能误解了这个问题。如果您想在应用程序中创建 Expose 事件,您可以创建并设置 Expose 事件,并使用XSendEvent发送它。

于 2013-06-10T17:40:46.443 回答