そのため、トップ レベルのウィンドウがいつ作成されるかを知る必要がある状況にいます。私は Xlib/Xt レベルで作業しており、EWMH 仕様をサポートしていないウィンドウ マネージャーを使用しています。私の考えは、ルート ウィンドウの SubstructureNotify イベントにフックすることです。しかし、物事はそれほど単純ではありません。
問題は、すべての CreateNotify イベントが [b]トップ レベル[/b] ウィンドウの作成に対応しているわけではないことです。だから私がする必要があると思うのは、イベントから取得したウィンドウをテストして、それがトップレベルのウィンドウであることを確認することです。私は近づいたが、いくつかの偽のウィンドウがまだ私のネットを通り抜けている. たとえば、GTK アプリケーションでは、ドロップダウン ボックスがあり、それをクリックすると、新しいウィンドウが作成されますが、それをキャッチして無視する方法がわかりません。このようなウィンドウは、典型的なトップ レベルのアプリケーション ウィンドウと見分けがつかないという厄介な問題を抱えています。
これが私がこれまでに持っているものです:
// I am omiting (tons of) cleanup code and where I set the display and toplevel variables.
Display* display;
Widget toplevel;
bool has_name(Window window)
{
XTextProperty data = XTextProperty ();
return (!XGetWMName (display, window, &data));
}
bool has_client_leader(Window window)
{
unsigned long nitems = 0;
unsigned char* data = 0;
Atom actual_type;
int actual_format;
unsigned long bytes;
// WM_CLIENT_LEADER is an interned Atom for the WM_CLIENT_LEADER property
int status = XGetWindowProperty (display, window, WM_CLIENT_LEADER, 0L, (~0L), False,
AnyPropertyType, &actual_type, &actual_format, &nitems, &bytes, &data);
if (status != Success || acutal_type == None) return false;
Window* leader = reinterpret_cast<Window*> (data);
return (*leader != 0);
}
bool has_class(Window window)
{
XClassHint data = XClassHint ();
return (!GetClassHint (display, window, &data));
}
void handle_event(Widget widget, XtPointer, XEvent* event, Boolean*)
{
if (event->type != CreateNotify) return;
Window w = event->xcreatewindow.window;
// confirm window has a name
if (!has_name (w)) return;
// confirm window is a client window
Window client = XmuClientWindow (display, w);
if (!client || client != w) return;
// confirm window has a client leader that is not 0x0
if (!has_client_leader (client)) return;
// confirm window has a class
if (!has_class (client)) return;
// The window has passed all our checks!
// Go on to do stuff with the window ...
}
int main(int argc, char* argv[])
{
// ...
// Setting up the event handler for SubstructureNotify on root window
Window root_window = XDefaultRootWindow (display);
Widget dummy = XtCreateWidget ("dummy", coreWidgetClass, toplevel, 0, 0);
XtRegisterDrawable (display, root_window, dummy);
XSelectInput (display, root_window, SubstructureNotifyMask);
XtAddRawEventHandler (dummy, SubstructureNotifyMask, False, handle_event, 0);
// ...
}
ロングショットですが、私が試すことができるアイデアはありますか? ここで本当にできることは他にあまり考えられません。