30

我正在制作一个与每个正在运行的应用程序交互的应用程序。现在,我需要一种获取窗口 z 顺序的方法。例如,如果 Firefox 和记事本正在运行,我需要知道哪个在前面。

有任何想法吗?除了为每个应用程序的主窗口执行此操作外,我还需要为其子窗口和姊妹窗口(属于同一进程的窗口)执行此操作。

4

5 回答 5

14

您可以使用 GetTopWindow 函数搜索父窗口的所有子窗口,并返回 z 顺序最高的子窗口的句柄。GetNextWindow 函数以 z 顺序检索下一个或上一个窗口的句柄。

GetTopWindow: http: //msdn.microsoft.com/en-us/library/ms633514
(VS.85).aspx GetNextWindow: http://msdn.microsoft.com/en-us/library/ms633509(VS.85 ) .aspx

于 2009-05-05T16:03:38.187 回答
11

简洁明了:

int GetZOrder(IntPtr hWnd)
{
    var z = 0;
    for (var h = hWnd; h != IntPtr.Zero; h = GetWindow(h, GW.HWNDPREV)) z++;
    return z;
}

如果您需要更高的可靠性:

/// <summary>
/// Gets the z-order for one or more windows atomically with respect to each other. In Windows, smaller z-order is higher. If the window is not top level, the z order is returned as -1. 
/// </summary>
int[] GetZOrder(params IntPtr[] hWnds)
{
    var z = new int[hWnds.Length];
    for (var i = 0; i < hWnds.Length; i++) z[i] = -1;

    var index = 0;
    var numRemaining = hWnds.Length;
    EnumWindows((wnd, param) =>
    {
        var searchIndex = Array.IndexOf(hWnds, wnd);
        if (searchIndex != -1)
        {
            z[searchIndex] = index;
            numRemaining--;
            if (numRemaining == 0) return false;
        }
        index++;
        return true;
    }, IntPtr.Zero);

    return z;
}

(根据备注部分GetWindowEnumChildWindowsGetWindow在循环中调用更安全,因为您的GetWindow循环对于外部更改不是原子的。根据参数部分EnumChildWindows,使用 null 父级调用等效于EnumWindows。)

然后,您无需为每个窗口单独调用EnumWindows(这也不是原子的并且不受并发更改的影响),而是将要比较的每个窗口发送到 params 数组中,以便可以同时检索它们的 z 顺序。

于 2012-03-02T05:11:32.153 回答
3

这是我的 C# 解决方案:该函数返回给定 HWND 的兄弟之间的 zIndex,从 0 开始,最低 zOrder。

using System;
using System.Runtime.InteropServices;

namespace Win32
{
    public static class HwndHelper
    {
        [DllImport("user32.dll")]
        private static extern IntPtr GetWindow(IntPtr hWnd, uint uCmd);

        public static bool GetWindowZOrder(IntPtr hwnd, out int zOrder)
        {
            const uint GW_HWNDPREV = 3;
            const uint GW_HWNDLAST = 1;

            var lowestHwnd = GetWindow(hwnd, GW_HWNDLAST);

            var z = 0;
            var hwndTmp = lowestHwnd;
            while (hwndTmp != IntPtr.Zero)
            {
                if (hwnd == hwndTmp)
                {
                    zOrder = z;
                    return true;
                }

                hwndTmp = GetWindow(hwndTmp, GW_HWNDPREV);
                z++;
            }

            zOrder = int.MinValue;
            return false;
        }
    }
}
于 2016-09-14T12:40:01.330 回答
2
            // Find z-order for window.
            Process[] procs = Process.GetProcessesByName("notepad");
            Process top = null;
            int topz = int.MaxValue;
            foreach (Process p in procs)
            {
                IntPtr handle = p.MainWindowHandle;
                int z = 0;
                do
                {
                    z++;
                    handle = GetWindow(handle, 3);
                } while(handle != IntPtr.Zero);

                if (z < topz)
                {
                    top = p;
                    topz = z;
                }
            }

            if(top != null)
                Debug.WriteLine(top.MainWindowTitle);
于 2010-12-29T23:39:36.543 回答
1

获取 Z-Order 实现此功能(通过使用GetWindowWindows API 功能)

[DllImport("user32.dll", SetLastError = true)]
static extern IntPtr GetWindow(IntPtr hWnd, uint uCmd);

static int GetWindowZOrder(IntPtr hWnd)
{
    var zOrder = -1;
    while ((hWnd = GetWindow(hWnd, 2 /* GW_HWNDNEXT */)) != IntPtr.Zero) zOrder++;
    return zOrder;
}

返回值是一个从零开始的索引。返回 -1 表示无效的 hWnd。

方法是通过窗户到顶部。爬升的总数是 Z-Order 的值。

于 2020-04-16T05:12:46.137 回答