从桌面快捷方式的快捷键运行C#程序时,GetForegroundWindow无法正常工作

时间:2010-04-21 16:04:35

标签: c# windows winforms keyboard-shortcuts foreground

我试图通过在我的应用程序表单的构造函数中调用本机GetForegroundWindow WinAPI函数来获取C#/ Net 2.0 / WinForms中前台窗口的句柄。

当我直接从Windows资源管理器或Total Commander运行程序时,它会正确识别Windows资源管理器或Total Commander窗口。

但是,如果我在桌面上创建程序的快捷方式并为快捷方式设置快捷键(比如说Ctrl + Alt + X),当我使用快捷方式运行程序时,前景窗口被标识为“Shell_TrayWnd窗口”(句柄0x00010064),而不是实际窗口。 (假设我正在运行Firefox,当我按下Ctrl + Alt + X我的程序启动并说前景窗口不是Firefox,它应该是,它说它是任务栏 - Shell_TrayWnd。)

    public MainForm()
    {
        this.InitializeComponent();

        IntPtr handle = WinAPI.GetForegroundWindow();
        this.Text = handle.ToString();
        StringBuilder title = new StringBuilder(255);
        if (WinAPI.GetWindowText(handle, title, 255) > 0)
        {
            this.Text += title.ToString();
        }
    }

如何获得真实的前景窗口?我是否也应该使用其他功能,如GetWindow?

谢谢

3 个答案:

答案 0 :(得分:1)

请注意,在您调用GetForegroundWindow时,任务栏可能是真正的前景窗口,只是因为它是资源管理器,它正在处理快捷方式,而任务栏属于资源管理器(Shell_TrayWnd是窗口)任务栏的类。)。

如果您想对全局活动窗口执行某些操作,最好启动应用程序并让它在后台等待。然后,您可以在应用程序运行时处理按键操作,因此Explorer不会干扰。

不知何故,这让我想起了Raymond Chen的一篇文章。

答案 1 :(得分:0)

我认为你正在尝试做同样的事情 - 在当前路径上从Explorer打开一个shell。

我遇到了完全相同的问题。这是一个适合我的程序。它使用EnumWindows搜索所有可见窗口,直到找到标题为真实路径的窗口。

using System;
using System.IO;
using System.Text;
using System.Diagnostics;
using System.Runtime.InteropServices;

public class ShellHere
{
    // Thanks to pinvoke.net for the WinAPI stuff

    [DllImport("user32.dll")]
    private static extern int EnumWindows(CallBackPtr callPtr, int lPar); 

    [DllImport("user32.dll")]
    static extern int GetWindowText(int hWnd, StringBuilder text, int count);

    [DllImport("user32.dll", EntryPoint="GetWindowLong")]
    private static extern IntPtr GetWindowLongPtr32(IntPtr hWnd, GWL nIndex);

    [DllImport("user32.dll", EntryPoint="GetWindowLongPtr")]
    private static extern IntPtr GetWindowLongPtr64(IntPtr hWnd, GWL nIndex);

    public delegate bool CallBackPtr(int hwnd, int lParam);
    private static CallBackPtr _callBackPtr;

    // This static method is required because Win32 does not support
    // GetWindowLongPtr directly
    public static IntPtr GetWindowLongPtr(IntPtr hWnd, GWL nIndex)
    {
        if (IntPtr.Size == 8)
            return GetWindowLongPtr64(hWnd, nIndex);
        else
            return GetWindowLongPtr32(hWnd, nIndex);
    }

    public static bool FindPathInTitle( int hwnd, int lparams )
    {
        const int nChars = 256;
        StringBuilder buffer = new StringBuilder( nChars );

        IntPtr result = GetWindowLongPtr( new IntPtr(hwnd), GWL.GWL_STYLE );

        // ignore invisible windows
        if ( (result.ToInt64() & WS_VISIBLE) != 0 )
        {
            if ( GetWindowText( hwnd, buffer, nChars ) > 0 )
            {
                string title = buffer.ToString();

                // ignore the taskbar
                if ( title.ToLower() != "start" && Directory.Exists( title ) )
                {
                    _folder = title;
                    return false;
                }
            }
        }

        return true;
    }

    private static string _folder;

    public static void Main()
    {
        _callBackPtr = new CallBackPtr( FindPathInTitle );
        EnumWindows( _callBackPtr, 0 );

        Process shell = new Process();
        shell.StartInfo.FileName = "cmd.exe";
        if ( !string.IsNullOrEmpty( _folder ) )
            shell.StartInfo.WorkingDirectory = _folder;
        shell.Start();
    }

    public enum GWL
    {
        GWL_WNDPROC =    (-4),
        GWL_HINSTANCE =  (-6),
        GWL_HWNDPARENT = (-8),
        GWL_STYLE =      (-16),
        GWL_EXSTYLE =    (-20),
        GWL_USERDATA =   (-21),
        GWL_ID =     (-12)
    }

    // Window Styles 
    const UInt32 WS_VISIBLE = 0x10000000;
}

到目前为止它对我有用(Win7-64)。请注意,您不必直接在资源管理器窗口中工作 - 它将使用Tab键顺序中的下一个。

答案 2 :(得分:0)

我不确定你需要前景窗口,所以这可能会有所帮助:

您可能会通过以下方式检测到您是通过快捷方式启动的:

在这种情况下,您可能会尝试获取Z顺序中的上一个窗口,或者在桌面窗口的顶部。