我的代码允许用户将鼠标悬停在控件上,然后它会响应。但是,我希望徘徊更快一点。有没有办法加快悬停反应?
答案 0 :(得分:5)
在SystemInformation.MouseHoverTime
事件生成之前有一个故意的时间延迟MouseHover
毫秒,作为一个简单的替代方案,您可以使用MouseEnter
事件代替,这将立即触发。
答案 1 :(得分:3)
可以。 Windows窗体中的悬停不遵循全局系统设置,并且已在dwHoverTime
字段中的NativeMethods.TRACKMOUSEEVENT
中设置为100毫秒。
然后在控件的本机窗口的WndProc
方法中,WM_MOUSEMOVE
被捕获来调用TrackMouseEvent
,这反过来导致WM_MOUSEHOVER
。您可以看到源代码here。
因此,您可以通过将鼠标悬停的所需超时设置为TrackMouseEvent
的WM_MOUSEMOVE
字段来处理dwHoverTime
并调用TRACKMOUSEEVENT
。同时处理WM_MOUSEHOVER
并引发一个自定义事件,例如MyMouseHover
。
然后,您可以轻松处理此自定义MyMouseHover
事件。
using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
public class SampleControl : Control
{
[DllImport("user32.dll")]
private static extern int TrackMouseEvent(ref TRACKMOUSEEVENT lpEventTrack);
[StructLayout(LayoutKind.Sequential)]
private struct TRACKMOUSEEVENT
{
public uint cbSize;
public uint dwFlags;
public IntPtr hwndTrack;
public uint dwHoverTime;
public static readonly TRACKMOUSEEVENT Empty;
}
private TRACKMOUSEEVENT track = TRACKMOUSEEVENT.Empty;
const int WM_MOUSEMOVE = 0x0200;
const int WM_MOUSEHOVER = 0x02A1;
const int TME_HOVER = 0x1;
const int TME_LEAVE = 0x2;
public event EventHandler MyMouseHover;
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_MOUSEMOVE)
{
track.hwndTrack = this.Handle;
track.cbSize = (uint)Marshal.SizeOf(track);
track.dwFlags = TME_HOVER | TME_LEAVE;
track.dwHoverTime = 500;
TrackMouseEvent(ref track);
}
if(m.Msg == WM_MOUSEHOVER)
{
MyMouseHover?.Invoke(this, EventArgs.Empty);
}
base.WndProc(ref m);
}
}
注释
系统范围的SystemInformation.MouseHoverTime
设置用于ToolTip
。但是MouseHover
事件紧随NativeMethods.TRACKMOUSEEVENT
的dwHoverTime
字段的值设置为100毫秒。
您可以在我的帖子中找到类似的实现,用于处理以下形式的标题栏上的悬停事件:Handle Mouse Hover on Titlebar of Form。
作为另一种选择,我还尝试设置控件的专用dwHoverTime
字段的trackMouseEvent
字段的值,并且该解决方案似乎也有效。
您可以找到代码here的VB版本。
答案 2 :(得分:2)
如果您为此目的使用ToolTip
组件,则可以将其InitialDelay
属性设置为小于默认值500(半秒)的值。
顺便说一下,AutoPopDelay
和ReshowDelay
属性也很有用,分别确定鼠标重新进入控件客户区时的显示时间和延迟。
答案 3 :(得分:2)
您可以尝试使用鼠标悬停事件 not 的替代方法,而不是this site上的第一个答案。否则,如果你改变它,它将影响所有应用程序,而不仅仅是你的应用程序。