我试图通过在我的应用程序表单的构造函数中调用本机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?
谢谢
答案 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)
我不确定你需要前景窗口,所以这可能会有所帮助:
您可能会通过以下方式检测到您是通过快捷方式启动的:
dwFlags
STARTF_TITLEISLINKNAME
在这种情况下,您可能会尝试获取Z顺序中的上一个窗口,或者在桌面窗口的顶部。