編輯:我是不寻找其他方法或更好的方法更改窗口的标题或向终端中的选项卡添加标题。建议重复 不要以任何方式几乎解决了为什么xdotool search …
才不是当标题设置为时拾取窗口标题控制端如上所述。
操作系统:Ubuntu 18.04
当我打开gnome 终端窗口,标题是dkb@dkb:~
我可以使用来改变它
wmctrl -r :ACTIVE: -N "NewName"
但是xdotool search …
,没有“看到”这个标题:
dkb@dkb:~$ xdotool search --name NewName
dkb@dkb:~$
我只是得到了提示,而不是提供了相应的窗口标识符。
另一方面,我可以使用工具本身设置标题然后xdotool search …
提供窗口标识符:
dkb@dkb:~$ xdotool getactivewindow set_window --name NewName
dkb@dkb:~$ xdotool search --name NewName
39845894
dkb@dkb:~$
换句话说,如果标题由工具,xdotool search
按应有的方式工作:
从man xdotool
搜索 [选项] 模式 使用正则表达式模式搜索具有标题、名称或类别的窗口。输出是以行分隔的 X 窗口标识符列表。
所以虽然这不是什么大问题,但我想知道原因是什么工具没有“看到”窗口标题设置wmctrl。
答案1
我们可以看到,它至少xdotool getactivewindow getwindowname
按预期运行。
$ wmctrl -r :ACTIVE: -N "Test1"
$ xdotool getactivewindow getwindowname
Test1
$ xdotool getactivewindow set_window --name Test2
$ xdotool getactivewindow getwindowname
Test2
让我们更深入地了解一下。我们可以使用xprop -id <id>
命令列出窗口的属性。对于由wmctrl
它设置名称的窗口,给出:
_NET_WM_OPAQUE_REGION(CARDINAL) = 0, 0, 1920, 995
_NET_STARTUP_ID(UTF8_STRING) = "brisk-menu-2514-mariusz-HP-Pavilion-Notebook-mate-terminal-1_TIME1314631"
WM_WINDOW_ROLE(STRING) = "mate-terminal-window-4442-66103825-1563868224"
_NET_WM_WINDOW_TYPE(ATOM) = _NET_WM_WINDOW_TYPE_NORMAL
_NET_WM_SYNC_REQUEST_COUNTER(CARDINAL) = 79691784, 79691785
_NET_WM_USER_TIME_WINDOW(WINDOW): window id # 0x4c00007
WM_CLIENT_LEADER(WINDOW): window id # 0x4c00001
_NET_WM_PID(CARDINAL) = 4442
WM_LOCALE_NAME(STRING) = "en_US.UTF-8"
WM_CLIENT_MACHINE(STRING) = "mariusz-HP-Pavilion-Notebook"
WM_NORMAL_HINTS(WM_SIZE_HINTS):
program specified minimum size: 345 by 141
program specified resize increment: 9 by 20
program specified base size: 16 by 30
window gravity: NorthWest
WM_PROTOCOLS(ATOM): protocols WM_DELETE_WINDOW, WM_TAKE_FOCUS, _NET_WM_PING, _NET_WM_SYNC_REQUEST
WM_CLASS(STRING) = "mate-terminal", "Mate-terminal"
_NET_WM_ICON_NAME(UTF8_STRING) = "Terminal"
_NET_WM_NAME(UTF8_STRING) = "Test1"
对于由其设置名称的窗口,xdotool
将给出:
_NET_WM_OPAQUE_REGION(CARDINAL) = 0, 0, 1920, 995
_NET_STARTUP_ID(UTF8_STRING) = "brisk-menu-2514-mariusz-HP-Pavilion-Notebook-mate-terminal-1_TIME1314631"
WM_WINDOW_ROLE(STRING) = "mate-terminal-window-4442-66103825-1563868224"
_NET_WM_WINDOW_TYPE(ATOM) = _NET_WM_WINDOW_TYPE_NORMAL
_NET_WM_SYNC_REQUEST_COUNTER(CARDINAL) = 79691784, 79691785
_NET_WM_USER_TIME_WINDOW(WINDOW): window id # 0x4c00007
WM_CLIENT_LEADER(WINDOW): window id # 0x4c00001
_NET_WM_PID(CARDINAL) = 4442
WM_LOCALE_NAME(STRING) = "en_US.UTF-8"
WM_CLIENT_MACHINE(STRING) = "mariusz-HP-Pavilion-Notebook"
WM_NORMAL_HINTS(WM_SIZE_HINTS):
program specified minimum size: 345 by 141
program specified resize increment: 9 by 20
program specified base size: 16 by 30
window gravity: NorthWest
WM_PROTOCOLS(ATOM): protocols WM_DELETE_WINDOW, WM_TAKE_FOCUS, _NET_WM_PING, _NET_WM_SYNC_REQUEST
WM_CLASS(STRING) = "mate-terminal", "Mate-terminal"
_NET_WM_ICON_NAME(UTF8_STRING) = "Terminal"
_NET_WM_NAME(STRING) = "Test2"
我们可以看到,在两种情况下都_NET_WM_NAME
设置正确。这是什么_NET_WM_NAME
?它是 x11 的扩展(关联,关联)
_NET_WM_名称
_NET_WM_NAME,UTF8_STRING
客户端应将其设置为窗口的 UTF-8 编码标题。如果设置,窗口管理器应优先使用此标题而不是 WM_NAME。
因此,x11 应用程序应该优先选择此属性WM_NAME
。让我们显示这两个属性。我已从这里:
#include <X11/Xlib.h>
#include <X11/Xatom.h>
#include <stdio.h>
#include <stdlib.h>
#define MAXSTR 1000
Display *display;
unsigned long window;
unsigned char *prop;
void check_status(int status, unsigned long window)
{
if (status == BadWindow) {
printf("window id # 0x%lx does not exists!", window);
exit(1);
}
if (status != Success) {
printf("XGetWindowProperty failed!");
exit(2);
}
}
unsigned char* get_string_property(char* property_name)
{
Atom actual_type, filter_atom;
int actual_format, status;
unsigned long nitems, bytes_after;
filter_atom = XInternAtom(display, property_name, True);
status = XGetWindowProperty(display, window, filter_atom, 0, MAXSTR, False, AnyPropertyType,
&actual_type, &actual_format, &nitems, &bytes_after, &prop);
check_status(status, window);
return prop;
}
unsigned long get_long_property(char* property_name)
{
get_string_property(property_name);
unsigned long long_property = prop[0] + (prop[1]<<8) + (prop[2]<<16) + (prop[3]<<24);
return long_property;
}
int main(int argc, char** argv)
{
char *display_name = NULL; // could be the value of $DISPLAY
display = XOpenDisplay(display_name);
if (display == NULL) {
fprintf (stderr, "%s: unable to open display '%s'\n", argv[0], XDisplayName (display_name));
}
int screen = XDefaultScreen(display);
window = RootWindow(display, screen);
window = get_long_property("_NET_ACTIVE_WINDOW");
printf("_NET_WM_PID: %lu\n", get_long_property("_NET_WM_PID"));
printf("WM_CLASS: %s\n", get_string_property("WM_CLASS"));
printf("_NET_WM_NAME: %s\n", get_string_property("_NET_WM_NAME"));
printf("WM_NAME: %s\n", get_string_property("WM_NAME"));
XCloseDisplay(display);
return 0;
}
仅添加一行printf("WM_NAME: %s\n", get_string_property("WM_NAME"));
。运行:
$ gcc test.c -o test -lX11
$ wmctrl -r :ACTIVE: -N "Test1"
$ ./test
_NET_WM_PID: 4442
WM_CLASS: mate-terminal
_NET_WM_NAME: Test1
WM_NAME: (null)
$ xdotool getactivewindow set_window --name Test2
$ ./test
_NET_WM_PID: 4442
WM_CLASS: mate-terminal
_NET_WM_NAME: Test2
WM_NAME: Test2
因此结论是:xdotool
同时设置_NET_WM_NAME
和WN_NAME
但仅通过WM_NAME
和wmctrl
设置进行搜索_NET_WM_NAME
。