通过调用SetForegroundWindow(IntPtr hWnd)和SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, SetWindowPosFlags uFlags),分别能实现窗口焦点的获取和窗口的置顶。在实现窗口的置顶过程中发现,如果当前焦点不在要置顶的窗口上,则置顶会失败。实现过程中使用GetForegroundWindow()获取当前焦点窗口的句柄,使用GetWindowRect(IntPtr hwnd, out RECT lpRect)获取置顶窗口的原始Rect,以保持置顶后窗口大小和位置不变。 代码如下:
public class SpecialWindow { [StructLayout(LayoutKind.Sequential)] public struct RECT { public int Left; // x position of upper-left corner public int Top; // y position of upper-left corner public int Right; // x position of lower-right corner public int Bottom; // y position of lower-right corner } [DllImport("user32.dll", SetLastError = true)] static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, SetWindowPosFlags uFlags); [DllImport("user32.dll", SetLastError = true)] static extern bool GetWindowRect(IntPtr hwnd, out RECT lpRect); [DllImport("user32.dll")] [return: MarshalAs(UnmanagedType.Bool)] static extern bool SetForegroundWindow(IntPtr hWnd); [DllImport("user32.dll")] static extern IntPtr GetForegroundWindow(); public static void SetWindowTopMost(IntPtr hWnd) { try { RECT rect; var v = GetWindowRect(hWnd, out rect); IntPtr currentFocusedWindowHandle = GetForegroundWindow(); if (currentFocusedWindowHandle != hWnd) { SetForegroundWindow(hWnd); } SetWindowPos(hWnd, (IntPtr)SpecialWindowHandles.HWND_TOP, rect.Left, rect.Top, rect.Right - rect.Left, rect.Bottom - rect.Top, SetWindowPosFlags.SWP_NOMOVE | SetWindowPosFlags.SWP_NOSIZE); } catch (Exception e) { Console.WriteLine(e); throw; } } } public enum SpecialWindowHandles { // ReSharper disable InconsistentNaming /// <summary> /// Places the window at the top of the Z order. /// </summary> HWND_TOP = 0, /// <summary> /// Places the window at the bottom of the Z order. If the hWnd parameter identifies a topmost window, the window loses its topmost status and is placed at the bottom of all other windows. /// </summary> HWND_BOTTOM = 1, /// <summary> /// Places the window above all non-topmost windows. The window maintains its topmost position even when it is deactivated. /// </summary> HWND_TOPMOST = -1, /// <summary> /// Places the window above all non-topmost windows (that is, behind all topmost windows). This flag has no effect if the window is already a non-topmost window. /// </summary> HWND_NOTOPMOST = -2 // ReSharper restore InconsistentNaming } [Flags] public enum SetWindowPosFlags : uint { // ReSharper disable InconsistentNaming /// <summary> /// If the calling thread and the thread that owns the window are attached to different input queues, the system posts the request to the thread that owns the window. This prevents the calling thread from blocking its execution while other threads process the request. /// </summary> SWP_ASYNCWINDOWPOS = 0x4000, /// <summary> /// Prevents generation of the WM_SYNCPAINT message. /// </summary> SWP_DEFERERASE = 0x2000, /// <summary> /// Draws a frame (defined in the window's class description) around the window. /// </summary> SWP_DRAWFRAME = 0x0020, /// <summary> /// Applies new frame styles set using the SetWindowLong function. Sends a WM_NCCALCSIZE message to the window, even if the window's size is not being changed. If this flag is not specified, WM_NCCALCSIZE is sent only when the window's size is being changed. /// </summary> SWP_FRAMECHANGED = 0x0020, /// <summary> /// Hides the window. /// </summary> SWP_HIDEWINDOW = 0x0080, /// <summary> /// Does not activate the window. If this flag is not set, the window is activated and moved to the top of either the topmost or non-topmost group (depending on the setting of the hWndInsertAfter parameter). /// </summary> SWP_NOACTIVATE = 0x0010, /// <summary> /// Discards the entire contents of the client area. If this flag is not specified, the valid contents of the client area are saved and copied back into the client area after the window is sized or repositioned. /// </summary> SWP_NOCOPYBITS = 0x0100, /// <summary> /// Retains the current position (ignores X and Y parameters). /// </summary> SWP_NOMOVE = 0x0002, /// <summary> /// Does not change the owner window's position in the Z order. /// </summary> SWP_NOOWNERZORDER = 0x0200, /// <summary> /// Does not redraw changes. If this flag is set, no repainting of any kind occurs. This applies to the client area, the nonclient area (including the title bar and scroll bars), and any part of the parent window uncovered as a result of the window being moved. When this flag is set, the application must explicitly invalidate or redraw any parts of the window and parent window that need redrawing. /// </summary> SWP_NOREDRAW = 0x0008, /// <summary> /// Same as the SWP_NOOWNERZORDER flag. /// </summary> SWP_NOREPOSITION = 0x0200, /// <summary> /// Prevents the window from receiving the WM_WINDOWPOSCHANGING message. /// </summary> SWP_NOSENDCHANGING = 0x0400, /// <summary> /// Retains the current size (ignores the cx and cy parameters). /// </summary> SWP_NOSIZE = 0x0001, /// <summary> /// Retains the current Z order (ignores the hWndInsertAfter parameter). /// </summary> SWP_NOZORDER = 0x0004, /// <summary> /// Displays the window. /// </summary> SWP_SHOWWINDOW = 0x0040, // ReSharper restore InconsistentNaming }通过调用Win32 SHGetPathFromIDList(IntPtr pidl, IntPtr pszPath)弹出选择文件夹窗口,并在窗口回调函数中获取窗口的句柄和实现窗口的置顶。代码如下:
public class BrowseForFolder { private IntPtr _currentIntPtr; private DispatcherTimer timer; private bool _timerStarted = false; // Constants for sending and receiving messages in BrowseCallBackProc public const int WM_USER = 0x400; public const int BFFM_INITIALIZED = 1; public const int BFFM_SELCHANGED = 2; public const int BFFM_VALIDATEFAILEDA = 3; public const int BFFM_VALIDATEFAILEDW = 4; public const int BFFM_IUNKNOWN = 5; // provides IUnknown to client. lParam: IUnknown* public const int BFFM_SETSTATUSTEXTA = WM_USER + 100; public const int BFFM_ENABLEOK = WM_USER + 101; public const int BFFM_SETSELECTIONA = WM_USER + 102; public const int BFFM_SETSELECTIONW = WM_USER + 103; public const int BFFM_SETSTATUSTEXTW = WM_USER + 104; public const int BFFM_SETOKTEXT = WM_USER + 105; // Unicode only public const int BFFM_SETEXPANDED = WM_USER + 106; // Unicode only // Browsing for directory. private uint BIF_RETURNONLYFSDIRS = 0x0001; // For finding a folder to start document searching private uint BIF_DONTGOBELOWDOMAIN = 0x0002; // For starting the Find Computer private uint BIF_STATUSTEXT = 0x0004; // Top of the dialog has 2 lines of text for BROWSEINFO.lpszTitle and one line if // this flag is set. Passing the message BFFM_SETSTATUSTEXTA to the hwnd can set the // rest of the text. This is not used with BIF_USENEWUI and BROWSEINFO.lpszTitle gets // all three lines of text. private uint BIF_RETURNFSANCESTORS = 0x0008; private uint BIF_EDITBOX = 0x0010; // Add an editbox to the dialog private uint BIF_VALIDATE = 0x0020; // insist on valid result (or CANCEL) private uint BIF_NEWDIALOGSTYLE = 0x0040; // Use the new dialog layout with the ability to resize // Caller needs to call OleInitialize() before using this API private uint BIF_USENEWUI = 0x0040 + 0x0010; //(BIF_NEWDIALOGSTYLE | BIF_EDITBOX); private uint BIF_BROWSEINCLUDEURLS = 0x0080; // Allow URLs to be displayed or entered. (Requires BIF_USENEWUI) private uint BIF_UAHINT = 0x0100; // Add a UA hint to the dialog, in place of the edit box. May not be combined with BIF_EDITBOX private uint BIF_NONEWFOLDERBUTTON = 0x0200; // Do not add the "New Folder" button to the dialog. Only applicable with BIF_NEWDIALOGSTYLE. private uint BIF_NOTRANSLATETARGETS = 0x0400; // don't traverse target as shortcut private uint BIF_BROWSEFORCOMPUTER = 0x1000; // Browsing for Computers. private uint BIF_BROWSEFORPRINTER = 0x2000;// Browsing for Printers private uint BIF_BROWSEINCLUDEFILES = 0x4000; // Browsing for Everything private uint BIF_SHAREABLE = 0x8000; // sharable resources displayed (remote shares, requires BIF_USENEWUI) [DllImport("shell32.dll")] static extern IntPtr SHBrowseForFolder(ref BROWSEINFO lpbi); // Note that the BROWSEINFO object's pszDisplayName only gives you the name of the folder. // To get the actual path, you need to parse the returned PIDL [DllImport("shell32.dll", CharSet = CharSet.Unicode)] // static extern uint SHGetPathFromIDList(IntPtr pidl, [MarshalAs(UnmanagedType.LPWStr)] //StringBuilder pszPath); static extern bool SHGetPathFromIDList(IntPtr pidl, IntPtr pszPath); [DllImport("user32.dll", PreserveSig = true)] public static extern IntPtr SendMessage(HandleRef hWnd, uint Msg, int wParam, IntPtr lParam); [DllImport("user32.dll", CharSet = CharSet.Auto)] public static extern IntPtr SendMessage(HandleRef hWnd, int msg, int wParam, string lParam); private string _initialPath; public delegate int BrowseCallBackProc(IntPtr hwnd, int msg, IntPtr lp, IntPtr wp); struct BROWSEINFO { public IntPtr hwndOwner; public IntPtr pidlRoot; public string pszDisplayName; public string lpszTitle; public uint ulFlags; public BrowseCallBackProc lpfn; public IntPtr lParam; public int iImage; } public int OnBrowseEvent(IntPtr hWnd, int msg, IntPtr lp, IntPtr lpData) { switch (msg) { case BFFM_INITIALIZED: // Required to set initialPath { //Win32.SendMessage(new HandleRef(null, hWnd), BFFM_SETSELECTIONA, 1, lpData); // Use BFFM_SETSELECTIONW if passing a Unicode string, i.e. native CLR Strings. SendMessage(new HandleRef(null, hWnd), BFFM_SETSELECTIONW, 1, _initialPath); _currentIntPtr = hWnd; //if (!_timerStarted) //{ // timer?.Start(); //} SpecialWindow.SetWindowTopMost(hWnd); break; } case BFFM_SELCHANGED: { IntPtr pathPtr = Marshal.AllocHGlobal((int)(260 * Marshal.SystemDefaultCharSize)); if (SHGetPathFromIDList(lp, pathPtr)) SendMessage(new HandleRef(null, hWnd), BFFM_SETSTATUSTEXTW, 0, pathPtr); Marshal.FreeHGlobal(pathPtr); //SpecialWindow.SetWindowTopMost(hWnd); break; } } return 0; } public string SelectFolder(string caption, string initialPath, IntPtr parentHandle) { timer = new DispatcherTimer(); timer.Interval = TimeSpan.FromSeconds(1); timer.Tick += TimerOnTick; _initialPath = initialPath; StringBuilder sb = new StringBuilder(256); IntPtr bufferAddress = Marshal.AllocHGlobal(256); ; IntPtr pidl = IntPtr.Zero; BROWSEINFO bi = new BROWSEINFO(); //bi.hwndOwner = parentHandle; bi.pidlRoot = IntPtr.Zero; bi.lpszTitle = caption; bi.ulFlags = BIF_NEWDIALOGSTYLE | BIF_SHAREABLE; bi.lpfn = new BrowseCallBackProc(OnBrowseEvent); bi.lParam = IntPtr.Zero; bi.iImage = 0; bi.pszDisplayName = "123123"; try { pidl = SHBrowseForFolder(ref bi); if (true != SHGetPathFromIDList(pidl, bufferAddress)) { return null; } sb.Append(Marshal.PtrToStringAuto(bufferAddress)); } finally { // Caller is responsible for freeing this memory. Marshal.FreeCoTaskMem(pidl); timer.Stop(); timer.Tick -= TimerOnTick; } return sb.ToString(); } private void TimerOnTick(object sender, EventArgs e) { SpecialWindow.SetWindowTopMost(_currentIntPtr); } }