问题
I want to close a System.Windows.Forms.Form if the user clicks anywhere outside it. I've tried using IMessageFilter, but even then none of the messages are passed to PreFilterMessage. How do I receive clicks outside a form's window?
回答1:
In your form's Deactivate event, put "this.Close()". Your form will close as soon as you click anywhere else in Windows.
Update: I think what you have right now is a Volume button, and inside the Click event you create an instance of your VolumeSlider form and make it appear by calling ShowDialog() which blocks until the user closes the popped-up form. In the next line you read the volume the user selected and use it in your program.
This is OK, but as you've noticed it forces the user to explicitly close the popup in order to get back to the main program. Show() is the method you really want to use here on your popup form, but Show() doesn't block which means the Click event back on your main form finishes without knowing what the new volume is supposed to be.
A simple solution is to create a public method on your main form like this:
public void SetVolume(int volume)
{
// do something with the volume - whatever you did before with it
}
Then, in your Volume button's Click event (also on the main form), you make the VolumeSlider appear like so:
VolumeSlider slider = new VolumeSlider();
slider.Show(this); // the "this" is needed for the next step
In the VolumeSlider form, as the user works the (I guess) scrollbar, you put this code in the scrollbar's ValueChanged event (I think that's what it is):
MainForm owner = (MainForm)this.Owner;
owner.SetVolume(scrollbar.Value);
And then in the VolumeSlider form's Deactivate event you would put this.Close() as mentioned above. Your form will then behave as expected.
回答2:
With thanks to p-daddy in this question, I've found this solution which allows me to use ShowDialog:
protected override void OnShown(EventArgs e)
{
base.OnShown(e);
this.Capture = true;
}
protected override void OnCaptureChanged(EventArgs e)
{
if (!this.Capture)
{
if (!this.RectangleToScreen(this.DisplayRectangle).Contains(Cursor.Position))
{
this.Close();
}
else
{
this.Capture = true;
}
}
base.OnCaptureChanged(e);
}
回答3:
With Simon's solution I had the same Problem describt by Noam. With following code I've avoid the "Click through" problem...
protected override void WndProc(ref Message m)
{
base.WndProc(ref m);
// if click outside dialog -> Close Dlg
if (m.Msg == NativeConstants.WM_NCACTIVATE) //0x86
{
if (this.Visible)
{
if (!this.RectangleToScreen(this.DisplayRectangle).Contains(Cursor.Position))
this.Close();
}
}
}
回答4:
If it is a child form in an MDI application, you could trap the click in the parent form, otherwise the solution will be messy.
I am not convinced what you suggest represents intuitive UI behaviour anyway. Are you sure that is the best design?
回答5:
If you are trying to make a popup window that behaves a little bit like a menu, except that it lets you interact with your controls, you could try hosting a usercontrol inside a toolstrip dropdown.
回答6:
SIMPLY WAY : on Form1 use this code to call form2:
Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click
Form2.Owner = Me
Form2.Show()
End Sub
and then use this code on form1 again :
Private Sub Form1_MouseClick(sender As Object, e As MouseEventArgs) Handles Me.MouseClick
If Form2.IsHandleCreated = True Then
Form2.Close()
End If
End Sub
回答7:
this is simple :
private void button1_Click(object sender, EventArgs e)
{
Form f = new Form();
f.LostFocus +=new EventHandler(f_LostFocus);
f.Show();
}
void f_LostFocus(object sender, EventArgs e)
{
Form f = sender as Form;
f.Close();
f.Dispose();
}
来源:https://stackoverflow.com/questions/298491/how-do-i-close-a-form-when-a-user-clicks-outside-the-forms-window