【发布时间】:2021-11-13 20:43:15
【问题描述】:
我想确定我的 XWindow 是最小化还是最大化。我的示例程序是:
/*
* Study for multiple windows.
*/
#include <X11/Xlib.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void prtxevtt(int type)
{
switch (type) {
case 2: fprintf(stderr, "KeyPress"); break;
case 3: fprintf(stderr, "KeyRelease"); break;
case 4: fprintf(stderr, "ButtonPress"); break;
case 5: fprintf(stderr, "ButtonRelease"); break;
case 6: fprintf(stderr, "MotionNotify"); break;
case 7: fprintf(stderr, "EnterNotify"); break;
case 8: fprintf(stderr, "LeaveNotify"); break;
case 9: fprintf(stderr, "FocusIn"); break;
case 10: fprintf(stderr, "FocusOut"); break;
case 11: fprintf(stderr, "KeymapNotify"); break;
case 12: fprintf(stderr, "Expose"); break;
case 13: fprintf(stderr, "GraphicsExpose"); break;
case 14: fprintf(stderr, "NoExpose"); break;
case 15: fprintf(stderr, "VisibilityNotify"); break;
case 16: fprintf(stderr, "CreateNotify"); break;
case 17: fprintf(stderr, "DestroyNotify"); break;
case 18: fprintf(stderr, "UnmapNotify"); break;
case 19: fprintf(stderr, "MapNotify"); break;
case 20: fprintf(stderr, "MapRequest"); break;
case 21: fprintf(stderr, "ReparentNotify"); break;
case 22: fprintf(stderr, "ConfigureNotify"); break;
case 23: fprintf(stderr, "ConfigureRequest"); break;
case 24: fprintf(stderr, "GravityNotify"); break;
case 25: fprintf(stderr, "ResizeRequest"); break;
case 26: fprintf(stderr, "CirculateNotify"); break;
case 27: fprintf(stderr, "CirculateRequest"); break;
case 28: fprintf(stderr, "PropertyNotify"); break;
case 29: fprintf(stderr, "SelectionClear"); break;
case 30: fprintf(stderr, "SelectionRequest"); break;
case 31: fprintf(stderr, "SelectionNotify"); break;
case 32: fprintf(stderr, "ColormapNotify"); break;
case 33: fprintf(stderr, "ClientMessage"); break;
case 34: fprintf(stderr, "MappingNotify"); break;
case 35: fprintf(stderr, "GenericEvent"); break;
default: fprintf(stderr, "???"); break;
}
}
int main(void) {
Window w;
GC gracxt;
XEvent e;
const char* msg = "Hello, window";
int s;
XFontStruct* font;
Display* d;
int front = 1;
Atom fullscreen;
int status;
Atom prop;
Atom type;
int format;
unsigned long length;
unsigned long after;
unsigned char* dp;
d = XOpenDisplay(NULL);
if (d == NULL) {
fprintf(stderr, "Cannot open display\n");
exit(1);
}
s = DefaultScreen(d);
w = XCreateSimpleWindow(d, RootWindow(d, s), 10, 10, 1000, 1000, 5,
BlackPixel(d, s), WhitePixel(d, s));
XSelectInput(d, w, ExposureMask|KeyPressMask|PointerMotionMask|
StructureNotifyMask|PropertyChangeMask);
XMapWindow(d, w);
gracxt = XCreateGC(d, w, 0, NULL);
font = XLoadQueryFont(d,
"-bitstream-courier 10 pitch-bold-r-normal--0-0-200-200-m-0-iso8859-1");
if (!font) {
fprintf(stderr, "*** No font ***\n");
exit(1);
}
XSetFont(d, gracxt, font->fid);
fullscreen = XInternAtom(d, "_NET_WM_STATE_FULLSCREEN", 1);
printf("Fullscreen atom: %ld\n", fullscreen);
while (1) {
XNextEvent(d, &e);
//printf("XWindow event: "); prtxevtt(e.type); printf("\n"); fflush(stdout);
if (e.type == Expose) XDrawString(d, e.xany.window, gracxt, 10, 50, msg, strlen(msg));
else if (e.type == ConfigureNotify) {
printf("ConfigureNotify: x: %d y: %d w: %d h: %d\n",
e.xconfigure.x, e.xconfigure.y, e.xconfigure.width, e.xconfigure.height);
} else if (e.type == PropertyNotify) {
//printf("PropertyNotify: name: %s value: %ld\n", XGetAtomName(d, e.xproperty.atom),
// e.xproperty.atom);
if (!strcmp(XGetAtomName(d, e.xproperty.atom), "_NET_WM_STATE")) {
status = XGetWindowProperty(d, w, e.xproperty.atom,
0L, 1L, 0,
AnyPropertyType, &type, &format,
&length, &after, &dp);
if (status == Success && dp && length) {
prop = ((Atom*)dp)[0];
printf("Property string: %s value: %ld\n", XGetAtomName(d, prop), prop);
}
}
}
XCloseDisplay(d);
return 0;
}
如果我最大化我得到的窗口:
ConfigureNotifyXWindow 事件: 配置通知:x:2 y:76 w:4976 h:2752 ExposeXWindow 事件:
这并没有真正告诉我用户最大化了它,只是它变大了。它与屏幕大小不匹配,这当然是正确的,因为它不包括桌面的标题和菜单栏。
点击最小化时,程序中根本没有任何指示。
在文档中:“Client to Window Manager Communication”
"4.2.5. 图标化和去图标化 未撤销的顶级窗口如果已映射则处于 Normal 状态,如果未映射则处于 Iconic 状态。即使窗口已重新设置父级,这也是正确的;切换到 Iconic 状态时,窗口管理器将取消映射窗口及其父窗口。 客户端可以通过选择顶级窗口上的 StructureNotify 事件来选择接收这些状态更改的通知。当它变为 Iconic 时会收到一个 UnmapNotify 事件,当它变为 Normal 时会收到一个 MapNotify 事件。”
我没有看到所描述的取消映射/映射通知行为。应该使用 StructureNotifyMask 启用此事件。
工作机器是带有 GDM3 的 Ubuntu 20.04。
谢谢,
斯科特·佛朗哥 加利福尼亚州圣何塞
通过 PropertyNotify 事件我得到:
PropertyNotifyXWindow event:
PropertyNotify: WM_STATE
PropertyNotifyXWindow event:
PropertyNotify: _NET_WM_STATE
PropertyNotifyXWindow event:
PropertyNotify: _GTK_EDGE_CONSTRAINTS
最小化/图标化。仍在努力获取原子的数据。
第二次尝试(使用上面的新代码):
PropertyNotifyXWindow event:
PropertyNotify: name: _NET_WM_STATE value: 326
Atom Property Value: a
PropertyNotifyXWindow event:
PropertyNotify: name: _GTK_EDGE_CONSTRAINTS value: 402
Atom Property Value: �
PropertyNotifyXWindow event:
PropertyNotify: name: _NET_WM_STATE value: 326
Atom Property Value: a
我在别处得到了 XGetWindowProperty() 示例。我不确定是什么 窗口管理器试图在这里告诉我。数据是另一个原子吗? (一个数字?)。
我从实用程序 xev 中找到了一些进一步的信息:
FocusIn event, serial 37, synthetic NO, window 0x4a00001,
mode NotifyNormal, detail NotifyNonlinear
KeymapNotify event, serial 37, synthetic NO, window 0x0,
keys: 70 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
PropertyNotify event, serial 37, synthetic NO, window 0x4a00001,
atom 0x146 (_NET_WM_STATE), time 4645675, state PropertyNewValue
PropertyNotify event, serial 37, synthetic NO, window 0x4a00001,
atom 0x192 (_GTK_EDGE_CONSTRAINTS), time 4645675, state PropertyNewValue
PropertyNotify event, serial 37, synthetic NO, window 0x4a00001,
atom 0x146 (_NET_WM_STATE), time 4645679, state PropertyNewValue
...
在测试窗口最大化时,最小化给出类似的跟踪。所以它说“state PropertyNewValue”,我正在寻找_NET_WM_STATE_FULLSCREEN。请问我如何找到新的属性值是什么?
结语:
我按照建议修改了上面的程序。我注释掉了不必要的打印。现在我明白了:
属性字符串:_NET_WM_STATE_MAXIMIZED_HORZ 值:333 配置通知:x:2 y:76 w:4976 h:2752 配置通知:x:20 y:90 w:1000 h:1000 属性字符串:_NET_WM_STATE_FOCUSED 值:353 属性字符串:_NET_WM_STATE_HIDDEN 值:330 属性字符串:_NET_WM_STATE_HIDDEN 值:330 属性字符串:_NET_WM_STATE_FOCUSED 值:353
在最大化窗口时收到 _NET_WM_STATE_MAXIMIZED_HORZ。当窗口恢复正常(从最小化或最大化)时收到 _NET_WM_STATE_FOCUSED 状态,并且 _NET_WM_STATE_HIDDEN 被图标化/最小化。
从互联网上的垃圾箱潜水中,我发现:
_NET_WM_STATE_MAXIMIZED_HORZ 要么 _NET_WM_STATE_MAXIMIZED_VERT
均值最大化。
_NET_WM_STATE_HIDDEN
似乎意味着图标化,但描述“表明如果其桌面/视口处于活动状态并且其坐标在屏幕范围内,则该窗口将不会在屏幕上可见”,是的。
_NET_WM_STATE_FOCUSED
"指示窗口的装饰是否在活动状态下绘制"
这似乎与“具有键盘焦点”的含义相同。由于您单击窗口以最小化/最大化窗口,因此猜测它可能是它的别名。
无论如何,谢谢大家的帮助,我现在有足够的信息。
斯科特·佛朗哥 加利福尼亚州圣何塞
PS。
观察到的一个小问题是,如果你最大化一个窗口,然后最小化它(不要恢复正常),然后选择图标,你会得到 每个事件的 _NET_WM_STATE_MAXIMIZED_HORZ,而不是您期望的 _NET_WM_STATE_HIDDEN。不知道这是一个错误还是什么。
S.
【问题讨论】:
-
对于符合 EWHM 的应用程序,您应该注意
PropertyNotify事件并查看_NET_WM_STATE属性是否已更改(更改为_NET_WM_STATE_FULLSCREEN等)。 -
不错。我会尝试并发布结果。
-
走得更远了,修改了源代码。我试图弄清楚如何从事件中获取数据(属性的值)。
-
我花了很多时间浏览 Xwindows Atoms 上的文档,我需要一些帮助。最新的程序在上面。
-
所以它肯定给了我 _NET_WM_STATE 上的“propertynewvalue”,但是请问如何获取这个更改后的属性的值?