【问题标题】:Click event not raised in expanded glass area?扩展玻璃区域中未引发点击事件?
【发布时间】:2011-12-16 18:12:06
【问题描述】:

所以我只是使用 DwmExtendFrameIntoClientArea(Vista / 7 Aero 的东西)将表单的玻璃区域扩展到客户区域。

我已经在重写 Form 类方法 OnMouseDown() 中发送了一条 Windows 消息,导致窗口可以随玻璃区域移动,如 Make a borderless form movable? 此处所述。

但是,正因为如此,当我点击展开的玻璃区域时,我没有收到任何表单 Click / MouseClick / DoubleClick 等事件。

当我双击顶部展开的玻璃区域时,我实际上希望表单最大化,就像普通标题栏一样。

Form-inheriting 类的代码如下:

protected override void OnMouseDown(MouseEventArgs e)
{
    // Fensterverschiebung in Glass-Regionen
    if (_glassMovable && e.Button == MouseButtons.Left
        && (e.X < _glassPadding.Left || e.X > Width - _glassPadding.Right
        || e.Y < _glassPadding.Top || e.Y > Height - _glassPadding.Bottom))
    {
        NativeMethods.ReleaseCapture();
        NativeMethods.SendMessage(Handle, NativeMethods.WM_NCLBUTTONDOWN,
            NativeMethods.HT_CAPTION, 0);
    }

    base.OnMouseDown(e);
}

protected override void OnMouseDoubleClick(MouseEventArgs e)
{
    // Fenstermaximierung / Minimierung in Glass-Regionen
    if (MaximizeBox && e.Button == MouseButtons.Left && e.Y < _glassPadding.Top)
    {
        if (WindowState == FormWindowState.Normal)
        {
            WindowState = FormWindowState.Maximized;
        }
        else if (WindowState == FormWindowState.Maximized)
        {
            WindowState = FormWindowState.Normal;
        }
    }

    base.OnMouseDoubleClick(e);
}

有什么方法可以让它工作吗?

【问题讨论】:

  • 您需要发布代码。我确实知道处理此问题的事件与点击最大化按钮的能力有关。如果它被禁用,那么双击标题栏将什么也不做,它实际上禁用了表单最大化的能力,至少在 WinForms 的演员中。
  • @Ramhound:好的,我添加了当前鼠标相关的代码。
  • 您在正确的轨道上发送HT_CAPTION 消息,但您不应该在 WinForms 事件处理程序中执行此操作。 Form 类有一个原生的WndProc() 过程,您应该为您的表单覆盖它以便发送HT_CAPTION。我不记得确切的代码了,但是I have a similar code sample for WPF
  • 你没有从那个答案中得到很好的建议。在搜索框中输入“wm_nchittest”。
  • @BoltClock:谢谢!这很有帮助。你知道我也可以通过右键获取窗口系统菜单吗?

标签: c# winforms winapi aero-glass


【解决方案1】:

指向 WPF 解决方案的 BoltClocks 链接启发了我以下类似 WinForms 的代码。

我现在覆盖 WndProc 而不是 OnMouse* 事件。

玻璃区域的行为与标题栏完全相同,例如在 Windows 7 中使用 Dock 支持拖放并双击以最大化/恢复。此外,此解决方案现在支持右键单击玻璃区域时的系统窗口上下文菜单。

[StructLayout(LayoutKind.Sequential)]
private struct RECT
{
    public int left;
    public int top;
    public int right;
    public int bottom;
}

private const int WM_NCHITTEST     = 0x0084;
private const int WM_NCRBUTTONDOWN = 0x00A4;
private const int WM_SYSCOMMAND    = 0x0112;
private const int HT_CAPTION       = 0x02;
private const int TPM_RIGHTBUTTON  = 0x0002;
private const int TPM_RETURNCMD    = 0x0100;

[DllImport("user32.dll")]
private static extern IntPtr GetSystemMenu(IntPtr hWnd, bool bRevert);

[DllImport("user32.dll")]
private static extern int TrackPopupMenu(int hMenu, int wFlags, int x, int y,
    int nReserved, int hwnd, ref RECT lprc);

[DllImport("user32.dll")]
private static extern int PostMessage(int hWnd, int Msg, int wParam,
    int lParam);

protected override void WndProc(ref Message m)
{
    if (!DesignMode)
    {
        switch (m.Msg)
        {
            case WM_NCHITTEST:
                if (MouseInClientArea())
                {
                    if (MouseInGlassArea())
                    {
                        m.Result = (IntPtr)HT_CAPTION;
                    }
                    else
                    {
                        m.Result = IntPtr.Zero;
                    }
                    return;
                }
                break;
            case WM_NCRBUTTONDOWN:
                if (MouseInClientArea())
                {
                    IntPtr menuHandle = GetSystemMenu(Handle, false);
                    RECT rect = new RECT();
                    int menuItem = TrackPopupMenu(menuHandle.ToInt32(),
                        TPM_RIGHTBUTTON | TPM_RETURNCMD,
                        Cursor.Position.X, Cursor.Position.Y, 0,
                        Handle.ToInt32(), ref rect);
                    if (menuItem != 0)
                    {
                        PostMessage(Handle.ToInt32(), WM_SYSCOMMAND,
                            menuItem, 0);
                    }
                }
                break;
        }
    }
    base.WndProc(ref m);
}

private bool MouseInClientArea()
{
    Point p = PointToClient(Cursor.Position);
    return (p.X > 0 && p.X < ClientRectangle.Width
        && p.Y > 0 && p.Y < ClientRectangle.Height);
}

private bool MouseInGlassArea()
{
    if (_glassPadding.Left == -1 || _glassPadding.Right == -1
        || _glassPadding.Top == -1 || _glassPadding.Bottom == -1)
    {
        return true;
    }
    else
    {
        Point p = PointToClient(Cursor.Position);
        return (p.X < _glassPadding.Left
            || p.X > ClientRectangle.Width - _glassPadding.Right
            || p.Y < _glassPadding.Top
            || p.Y > ClientRectangle.Height - _glassPadding.Bottom);
    }
}

【讨论】:

    猜你喜欢
    • 2012-12-18
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-09-30
    • 1970-01-01
    相关资源
    最近更新 更多