我有允许用户将鼠标悬停在控件上的代码,它会响应。但是,我希望悬停更快地开火。有没有办法加快悬停反应?
3 回答
SystemInformation.MouseHoverTime
在生成事件之前有一个故意的毫秒时间延迟MouseHover
,作为一个简单的替代方案,您可以使用MouseEnter
事件来代替,它将立即触发。
是的你可以。Windows 窗体中的悬停不遵循全局系统设置,并且已在字段中设置NativeMethods.TRACKMOUSEEVENT
为 100 毫秒。dwHoverTime
然后在WndProc
控件的本机窗口的方法中,WM_MOUSEMOVE
被困到调用TrackMouseEvent
这反过来又导致了一个WM_MOUSEHOVER
. 您可以在此处查看源代码。
因此,您可以通过将鼠标悬停所需的超时WM_MOUSEMOVE
设置为. 还可以处理和引发自定义事件,例如.TrackMouseEvent
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
事件遵循设置为 100 毫秒的dwHoverTime
字段值。NativeMethods.TRACKMOUSEEVENT
您可以在我的帖子中找到类似的实现,用于处理表单标题栏上的悬停事件:Handle Mouse Hover on Titlebar of Form。
作为另一种选择,我还尝试设置控件
dwHoverTime
私有trackMouseEvent
字段的字段值,并且该解决方案似乎也有效。你可以在这里找到 VB 版本的代码。
如果将此ToolTip
组件用于此目的,则可以将其InitialDelay
属性设置为小于默认值 500(半秒)的值。
顺便说一句,AutoPopDelay
和ReshowDelay
属性也很有用,分别确定鼠标重新进入控件客户区时的显示时间和延迟。