Открытие формы WinForms с TopMost = true, но не имея его украсть фокус?

У меня есть форма, которая всплывает на экране пользователя и TopMost=true, но это крадет фокус. Как я могу получить его не украсть фокус, когда он впервые появляется?

7 ответов


вставьте этот код в форму:

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

Это то, что работал для меня. Он обеспечивает самый верхний, но без фокусного воровства.

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

    private const int WS_EX_TOPMOST = 0x00000008;
    protected override CreateParams CreateParams
    {
       get
       {
          CreateParams createParams = base.CreateParams;
          createParams.ExStyle |= WS_EX_TOPMOST;
          return createParams;
       }
    }

Не забудьте опустить параметр верхний в Visual Studio designer или в другом месте.

это украдено, err, заимствовано, отсюда (нажмите на обходные пути):

https://connect.microsoft.com/VisualStudio/feedback/details/401311/showwithoutactivation-is-not-supported-with-topmost


вы можете сделать это так:

    private const int SW_SHOWNOACTIVATE = 4;
    private const int HWND_TOPMOST = -1;
    private const uint SWP_NOACTIVATE = 0x0010;

    [System.Runtime.InteropServices.DllImport("user32.dll", EntryPoint = "SetWindowPos")]
    private static extern bool SetWindowPos(
         int hWnd,             // Window handle
         int hWndInsertAfter,  // Placement-order handle
         int X,                // Horizontal position
         int Y,                // Vertical position
         int cx,               // Width
         int cy,               // Height
         uint uFlags);         // Window positioning flags

    [System.Runtime.InteropServices.DllImport("user32.dll")]
    private static extern bool ShowWindow(System.IntPtr hWnd, int nCmdShow);

    public static void ShowInactiveTopmost(System.Windows.Forms.Form frm)
    {
        try
        {
            ShowWindow(frm.Handle, SW_SHOWNOACTIVATE);
            SetWindowPos(frm.Handle.ToInt32(), HWND_TOPMOST,
            frm.Left, frm.Top, frm.Width, frm.Height,
            SWP_NOACTIVATE);
        }
        catch (System.Exception ex)
        {
            // error handling
        }
    }

я протестировал приведенный ниже код, используя таймер на form1 для создания экземпляра и отображения form2 с form1 в качестве владельца.

в показанном событии form2 я затем устанавливаю фокус на владельца, который является текущей активной формой.

у меня есть текстовое поле на form1 и смог продолжать писать в текстовом поле без потери фокуса во время этого процесса.

мой код таймера в form1:

private void timer1_Tick(object sender, EventArgs e)
{
    Form2 popup = new Form2();
    popup.TopMost = true;
    popup.Show(this);
    timer1.Enabled = false;
}

мой код в показанном событии form2:

private void Form2_Shown(object sender, EventArgs e)
{
    this.Owner.Focus();
}

Вы можете сделать это или просто установите TopMost в false и используйте переопределение ShowWithoutActivation, как заявил Ханс Пассант.

Edit: (или используйте P/invoke, как видно из дополнительного комментария Ханса Пассанта, который я пропустил, пока писал это)


я столкнулся с той же проблемой. Я не использую C#, а c++. Я думаю это может быть полезно в любом случае:

использование windows.h:

BOOL WINAPI SetWindowPos(
  __in      HWND hWnd,
  __in_opt  HWND hWndInsertAfter,
  __in      int X,
  __in      int Y,
  __in      int cx,
  __in      int cy,
  __in      UINT uFlags
);

передача флага SWP_NOACTIVATE аргументу uFlags работала для меня.


вы можете установить:

this.TopMost = True;

on Load событие этой формы.

со мной все в порядке!


вместо того, чтобы писать .setfocus()на _activated событие, запишите его в .shown событие формы.