获取桌面/ shell窗口的句柄

MFH*_*MFH 6 .net c# windows pinvoke winapi

在我的一个程序中,我需要测试用户当前是否正在关注桌面/ shell窗口.目前我正在使用user32.dll中的GetShellWindow()并将结果与​​GetForegroundWindow()进行比较.

这种方法有效,直到有人更改桌面壁纸,但是一旦壁纸被更改,GetShellWindow()的句柄就不再与GetForegroundWindow()中的句柄相匹配,我不明白为什么会这样.(操作系统: Windows 7 32位)

有没有更好的方法来检查桌面是否专注?如果用户更换壁纸,最好不会破坏?

编辑:我设计了一个解决方法:我正在测试句柄以生成类"SHELLDLL_DefView"的子句.如果有,则桌面处于焦点状态.虽然它在我的电脑上工作并不意味着它会一直工作......

tkl*_*zig 6

事情发生了一些变化,因为Windows 7中有可用作壁纸的幻灯片.您可以使用WorkerW,但这只适用于壁纸设置为幻灯片效果.

当将壁纸模式设置为幻灯片放映时,您必须搜索课程窗口WorkerW并检查孩子是否有SHELLDLL_DefView.如果没有幻灯片,您可以使用旧的GetShellWindow().

几个月前我遇到了同样的问题,我写了一个函数来获得正确的窗口.不幸的是我找不到它.但以下应该有效.仅缺少Win32 Imports:

public enum DesktopWindow
{
    ProgMan,
    SHELLDLL_DefViewParent,
    SHELLDLL_DefView,
    SysListView32
}

public static IntPtr GetDesktopWindow(DesktopWindow desktopWindow)
{
    IntPtr _ProgMan = GetShellWindow();
    IntPtr _SHELLDLL_DefViewParent = _ProgMan;
    IntPtr _SHELLDLL_DefView = FindWindowEx(_ProgMan, IntPtr.Zero, "SHELLDLL_DefView", null);
    IntPtr _SysListView32 = FindWindowEx(_SHELLDLL_DefView, IntPtr.Zero, "SysListView32", "FolderView");

    if (_SHELLDLL_DefView == IntPtr.Zero)
    {
        EnumWindows((hwnd, lParam) =>
        {
            if (GetClassName(hwnd) == "WorkerW")
            {
                IntPtr child = FindWindowEx(hwnd, IntPtr.Zero, "SHELLDLL_DefView", null);
                if (child != IntPtr.Zero)
                {
                    _SHELLDLL_DefViewParent = hwnd;
                    _SHELLDLL_DefView = child;
                    _SysListView32 = FindWindowEx(child, IntPtr.Zero, "SysListView32", "FolderView"); ;
                    return false;
                }
            }
            return true;
        }, IntPtr.Zero);
    }

    switch (desktopWindow)
    {
        case DesktopWindow.ProgMan:
            return _ProgMan;
        case DesktopWindow.SHELLDLL_DefViewParent:
            return _SHELLDLL_DefViewParent;
        case DesktopWindow.SHELLDLL_DefView:
            return _SHELLDLL_DefView;
        case DesktopWindow.SysListView32:
            return _SysListView32;
        default:
            return IntPtr.Zero;
    }
}
Run Code Online (Sandbox Code Playgroud)

在您的情况下,您将调用GetDesktopWindow(DesktopWindow.SHELLDLL_DefViewParent);以获取顶级窗口以检查它是否是前景窗口.


小智 5

这是GetClassName()用于检测桌面是否处于活动状态的解决方法:

  • Windows 首次启动时,桌面的 Class 为“Progman”
  • 更换壁纸后,桌面的Class为“WorkerW”

您可以针对这些进行测试以查看桌面是否已聚焦。

[DllImport("user32.dll")]
static extern int GetForegroundWindow();

[DllImport("user32.dll")]
static extern int GetClassName(int hWnd, StringBuilder lpClassName, int nMaxCount);

public void GetActiveWindow() {
    const int maxChars = 256;
    int handle = 0;
    StringBuilder className = new StringBuilder(maxChars);

    handle = GetForegroundWindow();

    if (GetClassName(handle, className, maxChars) > 0) {
        string cName = className.ToString();
        if (cName == "Progman" || cName == "WorkerW") {
            // desktop is active
        } else {
            // desktop is not active
        }
    }
}
Run Code Online (Sandbox Code Playgroud)