如何在仍然接收鼠标点击时阻止表单窃取焦点?

时间:2012-06-21 17:52:14

标签: c# winforms

我正在尝试在WinForms中创建自定义AutoCompleteTextBox控件。基本TextBox提供的自动完成功能仅使用(string).StartsWith而不是(string).Contains提供结果。

我在ListBox停靠在单独的Form中显示自动完成搜索结果。我可以阻止Form最初使用via:

来窃取焦点
protected override bool ShowWithoutActivation
{
   get { return true; }
}

然后,我可以通过覆盖Form方法来阻止WndProc完全获得焦点:

protected override void WndProc( ref Message m )
{
    base.WndProc( ref m );
    switch ( m.Msg )
    {
        case WM_MOUSEACTIVATE:
            m.Result = ( IntPtr ) MA_NOACTIVATEANDEAT;
            break;

        default:
            break;
    }

执行此操作时,ListBox中包含的Form将收到MouseMoved个事件,但不会收到MouseClicked个事件。

MA_NOACTIVATEANDEAT转换为MA_NOACTIVATE只会将鼠标事件传递给ListBox,但会导致ListBox点击以从{{1}中窃取焦点} Form位于 - 将其传递给ListBox所在的“浮动”Form

有什么方法可以防止“浮动”ListBox从“主要”Form开始关注,同时仍然在Form内收到MouseClick个事件?< / p>

3 个答案:

答案 0 :(得分:0)

将您的浮动表单转换为Usercontrol,如下所示: - 将ListBox拉到&amp;挂钩Click事件以模拟您的Focus场景。

using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace WindowsFormsApplication3
{
    public partial class InactiveForm : UserControl
    {
        private const int WS_EX_TOOLWINDOW = 0x00000080;
        private const int WS_EX_NOACTIVATE = 0x08000000;
        private const int WS_EX_TOPMOST = 0x00000008;


        [DllImport("user32")]
        public static extern int SetParent
         (IntPtr hWndChild, IntPtr hWndNewParent);

        [DllImport("user32")]
        public static extern int ShowWindow
         (IntPtr hWnd, int nCmdShow);


        protected override CreateParams CreateParams
        {
            get
            {

                CreateParams p = base.CreateParams;
                p.ExStyle |= (WS_EX_NOACTIVATE | WS_EX_TOOLWINDOW | WS_EX_TOPMOST);
                p.Parent = IntPtr.Zero;
                return p;
            }
        }

        public InactiveForm()
        {
            InitializeComponent();
        }

        public new void Show()
        {
            if (this.Handle == IntPtr.Zero) base.CreateControl();

            SetParent(base.Handle, IntPtr.Zero);
            ShowWindow(base.Handle, 1);
        }


        private void OnListBoxClicked(object sender, EventArgs e)
        {
            MessageBox.Show("Clicked List Box on floating control");

        }
    }
}

MainForm中的代码(附带一个按钮及其点击处理程序): -

这将使用ListBox调用浮动控件。

namespace WindowsFormsApplication3
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            InactiveForm f = new InactiveForm();
             f.Show();
        }
    }
}

当您将UserControl(无边框形式)显示为浮动表单时,您会注意到it wont receive focus even if you clicked (selected) any of its children。在这种情况下,子节点是usercontrol上的ListBox。

参考herehere

答案 1 :(得分:0)

除了Angshuman Agarwal之前的回答: 如果您希望当UserControl显示为浮动窗体时,主窗体不会停用,请将一些代码修改为UserControl:

private Control _mControl;

[DllImport("user32.dll")]
static extern IntPtr SendMessage(IntPtr hwnd, int wMsg, IntPtr wParam, IntPtr lParam);

public new void Show(Control c)
{
    if (c == null) throw new ArgumentNullException();
    _mControl = c;
    if (this.Handle == IntPtr.Zero) base.CreateControl();
    SetParent(base.Handle, IntPtr.Zero);
    ShowWindow(base.Handle, 1);
}

protected override void WndProc(ref Message m)
{
    if (m.Msg == 0x86)  //WM_NCACTIVATE
    {
        if (m.WParam != IntPtr.Zero) //activate
        {
            SendMessage(_mControl.Handle, 0x86, (IntPtr)1, IntPtr.Zero);
        }
        this.DefWndProc(ref m);
        return;
    }
    base.WndProc(ref m);
}

答案 2 :(得分:0)

我创造了类似的东西,我的解决方案与lAbstract建议的类似。 floatin表单有一个事件DoReturnFocus,它自定义TextBox(在这种情况下是AutoCompleteTextBox控件)订阅,并简单地将焦点设置回自身。为防止主窗体出现在浮动窗体的前面,我将主窗体设置为浮动窗体的所有者,因为拥有的窗体永远不会显示在其所有者窗体后面。

更多关于Form.Owner Property