60

Linux では、私の C++ アプリケーションは fork() と execv() を使用して OpenOffice の複数のインスタンスを起動し、いくつかの PowerPoint スライド ショーを表示しています。この部分は機能します。

次に、OpenOffice ウィンドウをディスプレイ上の特定の場所に移動できるようにしたいと考えています。XMoveResizeWindow() 関数でそれを行うことができますが、インスタンスごとに Window を見つける必要があります。

各インスタンスのプロセス ID を持っていますが、そこから X11 ウィンドウを見つけるにはどうすればよいですか?


更新- Andy の提案のおかげで、私はこれをやめました。ここにコードを投稿して、スタック オーバーフロー コミュニティと共有します。

残念ながら、Open Office は _NET_WM_PID プロパティを設定していないように見えるため、これで最終的に問題が解決するわけではありませんが、質問には答えます。

// Attempt to identify a window by name or attribute.
// by Adam Pierce <adam@doctort.org>

#include <X11/Xlib.h>
#include <X11/Xatom.h>
#include <iostream>
#include <list>

using namespace std;

class WindowsMatchingPid
{
public:
    WindowsMatchingPid(Display *display, Window wRoot, unsigned long pid)
        : _display(display)
        , _pid(pid)
    {
    // Get the PID property atom.
        _atomPID = XInternAtom(display, "_NET_WM_PID", True);
        if(_atomPID == None)
        {
            cout << "No such atom" << endl;
            return;
        }

        search(wRoot);
    }

    const list<Window> &result() const { return _result; }

private:
    unsigned long  _pid;
    Atom           _atomPID;
    Display       *_display;
    list<Window>   _result;

    void search(Window w)
    {
    // Get the PID for the current Window.
        Atom           type;
        int            format;
        unsigned long  nItems;
        unsigned long  bytesAfter;
        unsigned char *propPID = 0;
        if(Success == XGetWindowProperty(_display, w, _atomPID, 0, 1, False, XA_CARDINAL,
                                         &type, &format, &nItems, &bytesAfter, &propPID))
        {
            if(propPID != 0)
            {
            // If the PID matches, add this window to the result set.
                if(_pid == *((unsigned long *)propPID))
                    _result.push_back(w);

                XFree(propPID);
            }
        }

    // Recurse into child windows.
        Window    wRoot;
        Window    wParent;
        Window   *wChild;
        unsigned  nChildren;
        if(0 != XQueryTree(_display, w, &wRoot, &wParent, &wChild, &nChildren))
        {
            for(unsigned i = 0; i < nChildren; i++)
                search(wChild[i]);
        }
    }
};

int main(int argc, char **argv)
{
    if(argc < 2)
        return 1;

    int pid = atoi(argv[1]);
    cout << "Searching for windows associated with PID " << pid << endl;

// Start with the root window.
    Display *display = XOpenDisplay(0);

    WindowsMatchingPid match(display, XDefaultRootWindow(display), pid);

// Print the result.
    const list<Window> &result = match.result();
    for(list<Window>::const_iterator it = result.begin(); it != result.end(); it++)
        cout << "Window #" << (unsigned long)(*it) << endl;

    return 0;
}
4

8 に答える 8

25

これを行う唯一の方法は、探しているものが見つかるまでウィンドウのツリーをたどることです。トラバースは難しくありません (例が必要な場合は、xwininfo.c を見て xwininfo -root -tree が何をするかを確認してください)。

しかし、探しているウィンドウをどのように特定しますか? 一部のアプリケーションは、_NET_WM_PID というウィンドウ プロパティを設定します。

OpenOfficeは(ほとんどの Gnome アプリと同様に) そのプロパティを設定するアプリケーションの 1 つ だと私は信じています。

于 2008-09-30T02:27:54.990 に答える
15

/proc/PID/environ に WINDOWID という変数が含まれているかどうかを確認します

于 2010-05-02T18:42:26.280 に答える
12

パーティーに少し遅れた。ただし、2004年に、Harald Welteは、LD_PRELOADを介してXCreateWindow()呼び出しをラップし、プロセスIDを_NET_WM_PIDに格納するコードスニペットを投稿しました。これにより、作成された各ウィンドウにPIDエントリが含まれるようになります。

http://www.mail-archive.com/devel@xfree86.org/msg05806.html

于 2012-11-20T21:00:51.637 に答える
2

良い方法はありません。私が見る唯一の実際のオプションは次のとおりです。

  1. プロセスのアドレス空間を調べて、接続情報とウィンドウ ID を見つけることができます。
  2. netstat または lsof または ipcs を使用して接続を Xserver にマップし、(どういうわけか! 少なくとも root が必要です) 接続情報を調べてそれらを見つけることができます。
  3. インスタンスをスポーンするときは、別のウィンドウがマップされるまで待つことができます。
于 2008-09-30T02:20:57.347 に答える
1

各インスタンスのプロセス ID を把握していますか? OOo に関する私の経験では、OOo の 2 番目のインスタンスを実行しようとすると、単に OOo の最初のインスタンスと対話し、追加のファイルを開くように指示するだけです。

X のメッセージ送信機能を使用して、適切にウィンドウを要求する必要があると思います。OOo がそのカバーをどこかに文書化してくれることを願っています。

于 2008-09-30T02:13:55.430 に答える