如果您希望您的 应用程序中的窗口始终显示在不同 应用程序的窗口顶部,那么BringWindowToTop 函数绝对不是您想要的.对于初学者,正如您所注意到的,您必须使用计时器重复调用该函数。这应该是您的第一个线索,即它是错误的 API。另一个问题是它只是将您的窗口带到其进程的 Z 顺序顶部,而不是系统上运行的所有其他进程。正如documentation 解释的那样,
调用此函数类似于调用SetWindowPos 函数来更改窗口在Z 顺序中的位置。 BringWindowToTop 不会使窗口成为顶级窗口。
最后一句应该表明有更好的方法。 Windows 内置了对顶级窗口的支持(即那些应该总是出现在其他窗口之上的窗口):这些被称为 top-most 窗口。这正是你想要的。最顶层的窗口总是出现在非最顶层的窗口之上。
Raymond Chen 试图解释on his blog 的一些困惑。请注意,在这种情况下,HWND_TOP 等效于 BringWindowToTop。相反,你想要HWND_TOPMOST。
使窗口置顶的最简单方法是在创建窗口时指定WS_EX_TOPMOST 标志。 .NET Framework 将大部分窗口创建工作隐藏在幕后,但您可以在需要时通过覆盖表单类的 CreateParams property 来自定义参数。
这里有一些示例代码可以让表单始终处于最顶层:
Protected Overrides ReadOnly Property CreateParams() As CreateParams
Get
Const WS_EX_TOPMOST As Integer = &H00000008
Dim cp As CreateParams = MyBase.CreateParams
cp.ExStyle = cp.ExStyle Or WS_EX_TOPMOST
Return cp
End Get
End Property
如果您想在运行时切换窗口的最顶层状态,这将不起作用。为此,您将不得不 P/Invoke SetWindowPos 函数。 P/Invoke 与您在 VB6 中使用 Declare 语句所做的类似,但在 .NET 世界中语义发生了轻微变化——这就是为什么您不能在 VB.NET 中使用旧的 VB6 Declare 语句.
下面是 VB.NET 的代码:
<DllImport("user32.dll", SetLastError:=True)> _
Private Shared Function SetWindowPos(ByVal hWnd As IntPtr, ByVal hWndInsertAfter As IntPtr, ByVal X As Integer, ByVal Y As Integer, ByVal cx As Integer, ByVal cy As Integer, ByVal uFlags As Integer) As Boolean
End Function
Private Const SWP_NOSIZE As Integer = &H1
Private Const SWP_NOMOVE As Integer = &H2
Private Shared ReadOnly HWND_TOPMOST As New IntPtr(-1)
Private Shared ReadOnly HWND_NOTOPMOST As New IntPtr(-2)
Public Function MakeTopMost()
SetWindowPos(Me.Handle(), HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE Or SWP_NOSIZE)
End Function
Public Function MakeNormal()
SetWindowPos(Me.Handle(), HWND_NOTOPMOST, 0, 0, 0, 0, SWP_NOMOVE Or SWP_NOSIZE)
End Function