как выйти из блокировки XNextEvent xlib
в windows поток GUI обычно вызывает GetMessage для ожидания сообщения, когда другой поток использует PoseMessage поместить сообщение в очередь, то Поток GUI вернет GetMessage (прекратите блокировку).
кто-нибудь может сказать мне, когда я использую XNextEvent под XWindows для ожидания событие, как я могу "разбудить" поток GUI в другом потоке. Есть API как PoseMessage я могу использовать ?.
3 ответов
нет. Вот почему большинство UI-фреймворков (Gtk, KDE и т. д.) используют пользовательские основные циклы для прослушивания большего количества источников событий.
внутренне XNextEvent использует сокет, поэтому он вызывает select()
чтобы знать, когда вход. Так что вы можете: Call ConnectionNumber(display)
чтобы получить дескриптор файла, который вам нужно пройти select()
это позволяет прослушивать несколько файловых дескрипторов.
пример кода из http://www.linuxquestions.org/questions/showthread.php?p=2431345#post2431345
#include <stdio.h>
#include <stdlib.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
Display *dis;
Window win;
int x11_fd;
fd_set in_fds;
struct timeval tv;
XEvent ev;
int main() {
dis = XOpenDisplay(NULL);
win = XCreateSimpleWindow(dis, RootWindow(dis, 0), 1, 1, 256, 256, \
0, BlackPixel (dis, 0), BlackPixel(dis, 0));
// You don't need all of these. Make the mask as you normally would.
XSelectInput(dis, win,
ExposureMask | KeyPressMask | KeyReleaseMask | PointerMotionMask |
ButtonPressMask | ButtonReleaseMask | StructureNotifyMask
);
XMapWindow(dis, win);
XFlush(dis);
// This returns the FD of the X11 display (or something like that)
x11_fd = ConnectionNumber(dis);
// Main loop
while(1) {
// Create a File Description Set containing x11_fd
FD_ZERO(&in_fds);
FD_SET(x11_fd, &in_fds);
// Set our timer. One second sounds good.
tv.tv_usec = 0;
tv.tv_sec = 1;
// Wait for X Event or a Timer
int num_ready_fds = select(x11_fd + 1, &in_fds, NULL, NULL, &tv);
if (num_ready_fds > 0)
printf("Event Received!\n");
else if (num_ready_fds == 0)
// Handle timer here
printf("Timer Fired!\n");
else
printf("An error occured!\n");
// Handle XEvents and flush the input
while(XPending(dis))
XNextEvent(dis, &ev);
}
return(0);
}
вы можете выйти из блокировки XNextEvent, отправив себе фиктивное событие.
Window interClientCommunicationWindow;
Bool x11EventLoopActive = True;
// create a dummy window, that we can use to end the blocking XNextEvent call
interClientCommunicationWindow = XCreateSimpleWindow(dpy, root, 10, 10, 10, 10, 0, 0, 0);
XSelectInput(dpy, interClientCommunicationWindow, StructureNotifyMask);
XEvent event;
while(x11EventLoopActive) {
XNextEvent(dpy, &event);
...
}
в другом потоке вы можете сделать это, чтобы закончить цикл:
x11EventLoopActive = False;
// push a dummy event into the queue so that the event loop has a chance to stop
XClientMessageEvent dummyEvent;
memset(&dummyEvent, 0, sizeof(XClientMessageEvent));
dummyEvent.type = ClientMessage;
dummyEvent.window = interClientCommunicationWindow;
dummyEvent.format = 32;
XSendEvent(dpy, interClientCommunicationWindow, 0, 0, (XEvent*)&dummyEvent);
XFlush(dpy);
вы должны использовать : Bool XCheckMaskEvent (дисплей*, длинный, XEvent)
на XCheckMaskEvent
функция сначала ищет очередь событий, а затем все события, доступные в соединении с сервером для первого события, которое соответствует указанной маске.
если он найдет совпадение,XCheckMaskEvent
удаляет это событие, копирует его в указанное XEvent
структура и возвращает True
. Другие события, хранящиеся в очереди, не отбрасываются.
если запрошенное вами событие недоступно,XCheckMaskEvent
возвращает False
, и выходной буфер будет сброшен.