4

每当用户将鼠标悬停在菜单项上时,我都需要显示一个无模式的消息框。我不能使用 messagebox.show(...) 因为它是一种模式。所以我所做的是创建一个单独的窗口表单并使用菜单项上的悬停事件显示表单。我有两个问题:

1)当窗口窗体显示菜单失去其可见性。
2) windows 窗体不会像工具提示那样出现在菜单项旁边。

关于如何自定义组件的工具提示以使其看起来和行为像 Windows 窗体的任何想法?

4

2 回答 2

5

要回答您的第二个问题:

如果将form.StartPosition属性设置为,FormStartPosition.Manual则可以将表单定位在光标处(例如):

form.StartPosition = FormStartPosition.Manual;
form.Location = new Point(Cursor.Position.X - 1, Cursor.Position.Y - 1);

这也可能有助于解决您的第一个问题。

如果您希望表单的行为类似于工具提示,那么如果您添加以下事件处理程序代码,它可能会满足您的需求:

    private void Form_MouseLeave(object sender, EventArgs e)
    {
        // Only close if cursor actually outside the popup and not over a label
        if (Cursor.Position.X < Location.X || Cursor.Position.Y < Location.Y ||
            Cursor.Position.X > Location.X + Width - 1 || Cursor.Position.Y > Location.Y + Height - 1)
        {
            Close();
        }
    }

这解释了-1设置表单位置。它确保光标在第一次显示时实际上位于表单上。

于 2009-06-19T12:03:15.070 回答
5

由于 Form 类只是原生窗口的包装,您可以使用以下代码段创建自己的弹出表单,它看起来几乎就像工具提示窗口:

public class PopupForm : Form
{
    private const int SWP_NOSIZE = 0x0001;
    private const int SWP_NOMOVE = 0x0002;
    private const int SWP_NOACTIVATE = 0x0010;

    private const int WS_POPUP = unchecked((int)0x80000000);
    private const int WS_BORDER = 0x00800000;

    private const int WS_EX_TOPMOST = 0x00000008;
    private const int WS_EX_NOACTIVATE = 0x08000000;

    private const int CS_DROPSHADOW = 0x00020000;

    private static readonly IntPtr HWND_TOPMOST = (IntPtr)(-1);

    [DllImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    private static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, uint uFlags);

    public PopupForm()
    {
        InitializeComponent();
        SetStyle(ControlStyles.Selectable, false);
        Visible = false;
    }

    protected virtual void InitializeComponent()
    {
        FormBorderStyle = FormBorderStyle.None;
        StartPosition = FormStartPosition.Manual;
        ShowInTaskbar = false;
        BackColor = SystemColors.Info;

        // ...
    }

    protected override CreateParams CreateParams
    {
        get
        {
            CreateParams cp = base.CreateParams;
            cp.Style |= WS_POPUP;
            cp.Style |= WS_BORDER;
            cp.ExStyle |= WS_EX_TOPMOST | WS_EX_NOACTIVATE;
            //if (Microsoft.OS.IsWinXP && SystemInformation.IsDropShadowEnabled)
            //    cp.ClassStyle |= CS_DROPSHADOW;
            return cp;
        }
    }

    protected override bool ShowWithoutActivation
    {
        get { return true; }
    }

    public new void Show()
    {
        SetWindowPos(Handle, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOSIZE | SWP_NOMOVE);    
        base.Show();
    }

    public void Show(Point p)
    {
        Location = p;
        Show();
    }
}

使用外部代码中的 Show() 和 Hide() 方法控制此表单。

于 2009-06-21T09:59:39.247 回答