Clique em evento não gerado na área do vidro expandido?

então eu apenas expandi a área de vidro do meu formulário para a área do cliente com DwmExtendFrameIntoClientArea (material Vista/7 Aero).

Já enviei uma mensagem do Windows no método Override Form class OnMouseDown() fazendo com que a janela seja movida com a área do vidro, conforme explicado aqui Transformar uma forma sem fronteiras? .

No entanto, por causa disso, não estou recebendo eventos de Click/MouseClick/DoubleClick etc. ao clicar na área do vidro expandido.

Na verdade, quero que o formulário seja maximizado quando clico duas vezes na área superior de vidro expandido, como ocorre com as barras de título normais.

Aqui está o código da classe herdada do formulário:

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);
}

Existe alguma maneira de fazer isso funcionar?

1
Você está no caminho certo ao enviar mensagens HT_CAPTION , mas você não deveria estar fazendo isso em um manipulador de eventos WinForms. A classe Form tem um procedimento nativo WndProc() que você deve sobrescrever para o seu formulário para enviar HT_CAPTION . Não consigo me lembrar do código exato, mas Eu tenho um exemplo de código semelhante para o wpf .
adicionado o autor BoltClock, fonte
@ Pac-Man: Que eu não tenho certeza, desculpe: /
adicionado o autor BoltClock, fonte
Você não recebeu conselhos muito bons dessa resposta. Digite 'wm_nchittest' na caixa Pesquisar.
adicionado o autor Hans Passant, fonte
Você precisará postar código. Eu sei que o evento que lida com isso está ligado à capacidade de pressionar o botão Maximizar. Se isso estiver desabilitado, clique duas vezes na barra de título para não fazer nada, isso realmente desabilita a capacidade do formulário ser maximizado, pelo menos no elenco do WinForms.
adicionado o autor Security Hound, fonte
@Ramhound: Ok, eu adicionei o código atual do mouse.
adicionado o autor Ray Koopa, fonte
@BoltClock: Obrigado! Isso ajudou muito. Você sabe como eu posso obter o menu do sistema de janela através do botão direito também?
adicionado o autor Ray Koopa, fonte
@BoltClock: Descobri como e adicionei na solução abaixo (que acabei de aceitar como resposta) :)
adicionado o autor Ray Koopa, fonte

1 Respostas

BoltClocks link para uma solução para o wpf me inspirou para o seguinte código semelhante para WinForms.

Eu agora substituo o WndProc em vez dos eventos OnMouse *.

A região do vidro se comporta exatamente como uma barra de título, e. Arrastar e soltar com suporte a dock no Windows 7 e clique duas vezes para maximizar/restaurar. Além disso, essa solução agora suporta o menu de contexto da janela do sistema ao clicar com o botão direito do mouse na área do vidro.

[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);
    }
}
2
adicionado