Обрабатывать события «нового окна верхнего уровня» в Xlib / Xt - PullRequest
4 голосов
/ 10 апреля 2009

Так что я нахожусь в ситуации, когда мне нужно знать, когда создается окно верхнего уровня. Я работаю на уровне Xlib / Xt и в Window Manager, который не поддерживает спецификацию 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);

// ...
}

В общем, но у кого-нибудь есть идеи, которые я мог бы попробовать? Я не могу думать о многом другом, что я действительно могу сделать здесь.

1 Ответ

2 голосов
/ 22 апреля 2009

Полагаю, вы знакомы с ICCCM и его многословным обсуждением .

Вы проверили свойство WM_TRANSIENT_FOR?

...