通过避免文本框验证关闭C#windows窗体

通过避免文本框验证关闭C#windows窗体,c#,textbox,C#,Textbox,这是一个winform C#问题。我有一个带有验证事件侦听器的文本框,用于根据正则表达式验证文本框的内容 验证后,如果输入的值不正确,我将显示messagebox并取消事件,以便鼠标光标移回具有不正确值的textbox 当我从该文本框移到其他按钮/文本框时,这工作正常 但是,当我输入不正确的值并关闭表单(右上角有关闭按钮)时,它会验证textbox内容并抛出messagebox,而表单在我调用事件时不会关闭 问题是,当我单击表单右上角的X按钮时,我不希望触发验证,因为我正在关闭表单。我该怎么做

这是一个winform C#问题。我有一个带有验证事件侦听器的文本框,用于根据正则表达式验证文本框的内容

验证后,如果输入的值不正确,我将显示messagebox并取消事件,以便鼠标光标移回具有不正确值的textbox

当我从该文本框移到其他按钮/文本框时,这工作正常

但是,当我输入不正确的值并关闭表单(右上角有关闭按钮)时,它会验证textbox内容并抛出messagebox,而表单在我调用事件时不会关闭

问题是,当我单击表单右上角的X按钮时,我不希望触发验证,因为我正在关闭表单。我该怎么做

我将尽快发布代码片段

private void Form1_FormClosing(object sender, FormClosingEventArgs e) {
    // Assume that X has been clicked and act accordingly.
}

创建一个关闭事件,然后简单地取消验证器。

尝试将
CauseValidation
设置为false

或者请看这里:

或者尝试在formClosing事件中设置此选项

private void Form1_Closing(object sender, System.ComponentModel.CancelEventArgs e)
{
    // CauseValidation to false  or check 

}

要使用下面的“txtIPAddress_validating()”等验证处理程序来关闭表单,而不必输入有效条目,我需要执行以下操作:

1) 初始化验证处理程序: 从要激活验证的控件的控件属性中,双击此控件事件列表中的“验证”事件。通过单击此控件的属性页的“事件(看起来像闪电的)”工具栏按钮,可以访问控件事件列表。然后,您可以在自动生成的处理程序中输入代码,并使用一个结合控件名称和“\u验证”的名称。此处理程序中某些内容被确定为无效的部分可以通过添加“e.Cancel=true”指令强制输入有效条目

有关此类验证方法示例,请参阅下面的“txtIPAddress_validating()”和“txtMaskBits_validating()”代码。不要因为这些特定示例的完整验证机制而分心。为了强制验证,您只需在自己的代码中看到并重现“e.Cancel=true”指令,并将其添加到您自己的验证方法的正确位置。此时该值被标识为无效

此时,验证应该可以正常工作,但任何关闭表单的尝试都将触发验证,在能够关闭表单之前,验证将停止并坚持有效值。这并不总是你想要的。如果情况并非如此,我将继续以下内容

2) 同时取消(禁用)所有验证的“取消”按钮:

a) 在窗体上放置一个常规的“取消”按钮,该按钮与下面的“btnCancel_Click()”方法关联

b) 在常规的“this.close();”之前指令,添加“AutoValidate=AutoValidate.Disable;”指示此指令禁用所有“验证”触发器。请注意,“btnCancel_Click”事件是在进行任何验证之前触发的。对于验证事件后将全部执行的表单关闭事件,情况并非如此。这就是为什么不能从这些表单关闭事件中禁用验证

c) 要使此“取消”按钮正常工作,还需要将此“取消”按钮的“原因验证”属性设置为“false”。这是必要的,否则单击此按钮将触发验证前的验证,可通过上述“AutoValidate=AutoValidate.Disable;”禁用验证指示

此时,您应该可以通过单击“取消”按钮退出,而无需首先输入有效值。但是,单击表单窗口右上角的“X”按钮仍将强制验证

3) 使右上角的“X”按钮也取消验证:

这里的挑战是在执行验证之前捕获这样的“X”点击事件。任何通过表单关闭处理程序执行的尝试都不会起作用,因为一旦执行到达该处理程序,就太晚了。但是,通过重写WndProc()方法并测试'm.Msg==0x10'条件,可以立即捕获“X”按钮的点击。如果该条件为true,则前面引入的“AutoValidate=AutoValidate.Disable;”在这种情况下,指令也可以用于禁用整体验证。有关此类方法的代码示例,请参见下面的WndProc()方法。您应该能够像在窗体类中一样复制和粘贴该方法

此时,“取消”和“X”按钮都应取消价值。但是,可用于关闭窗体的escape键不可用。当使用窗体的“CancelButton”属性将此转义键链接到窗体的“Cancel”按钮时,将激活此转义键

4) 使退出键也取消验证:

与“X”按钮类似,可以通过覆盖现有方法来捕获转义键。这就是ProcessDialogKey()方法。再一次,前面引入的'AutoValidate=AutoValidate.Disable;'指令也可用于禁用转义键的整体验证。请参阅下面代码中的“ProcessDialogKey()”重写方法,以了解如何执行此操作。在这里,您应该能够像在自己表单的类中一样复制和粘贴该方法

在这一点上,你应该做

进一步考虑:

值得注意的是,以下两种关闭窗口的其他方法此时也可以正常工作。这两种方式是:

  • 左上角窗口图标按钮的“关闭”选项
  • 按Alt+F4,触发与上述“关闭”选项相同的关闭操作
一旦您引入了上面第3点中描述的“X”按钮捕获机制,这两种关闭窗口的方法也开始取消验证

到目前为止,这就是我想要的。希望这有帮助!
public partial class frmMyIP : Form
{
    public frmMyIP()
    {
          InitializeComponent();
    }

    // To capture the Upper right "X" click
    protected override void WndProc(ref Message m)
    {
        if (m.Msg == 0x10) // The upper right "X" was clicked
        {
            AutoValidate = AutoValidate.Disable; //Deactivate all validations
        }
        base.WndProc(ref m);
    }

    // To capture the "Esc" key
    protected override bool ProcessDialogKey(Keys keyData)
    {
        if (keyData == Keys.Escape)
        {
            AutoValidate = AutoValidate.Disable;
            btnCancel.PerformClick(); 
            return true;
        }
        return base.ProcessDialogKey(keyData);
    }
    public bool IsValidIP(string ipaddr)
    {
        string pattern = @"^([1-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])"+
        @"(\.([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])){3}$";

        Regex check = new Regex(pattern);
        bool valid = false;

        if (ipaddr == "")
        {
            valid = false;
        }
        else
        {
            valid = check.IsMatch(ipaddr, 0);
        }

        return valid;
    }

    private void txtIPAddress_Validating(object sender, CancelEventArgs e)
    {
        string address = txtIPAddress.Text;
        if (!IsValidIP(address))
        {
            MessageBox.Show("Invalid IP address!");
            e.Cancel = true;
        }
    }

    private void cmbMaskBits_Validating(object sender, CancelEventArgs e)
    {
        int MaskBitsValue = Convert.ToInt32(cmbMaskBits.Text);

        if (MaskBitsValue<1 || MaskBitsValue>30)
        {
        MessageBox.Show("Please select a 'Mask Bits' value between 1 and 30!");
        e.Cancel = true;
        }
    }

    private void btnCancel_Click(object sender, EventArgs e)
    {
        // Stop the validation of any controls so the form can close.
        // Note: The CausesValidation property of this <Cancel> button
        //       must also be set to false.

        AutoValidate = AutoValidate.Disable;
        this.Close();
    }
//Allow the form to be closed
if (this.ActiveControl.Equals(sender))
    return;
    public Form1()
    {
        // Disable validation in constructor
        textBox.CausesValidation = false;
    }

    private void OnSaveClicked(object sender, EventArgs e)
    {
        textBox.CausesValidation = true;
        if (ValidateChildren())
        {
            //
            // Do saving of the form data or other processing here ....
            //
            Close();
        }
        //
        // Set validation to false, as user may press Cancel next
        //
        textBox.CausesValidation = false;
    }

    private void OnCancelClicked(object sender, EventArgs e)
    {
        Close();
    }
Private Sub validateCell(ByVal tagDesc As String, ByVal userInput As String, ByVal legalRegex As String, ByVal regexDesc As String, ByVal e As DataGridViewCellValidatingEventArgs)
    Dim match As Match = Regex.Match(userInput, legalRegex)
    Dim matches = match.Groups()
    Dim val = match.Value
    If val.Length = 0 Or userInput.Length > val.Length Then
        tagGrid.Rows(e.RowIndex).ErrorText = _
            tagDesc & " must match pattern:  " & regexDesc
        If Me.Cancel_Button.Focused Or Me.clearButton.Focused Then
            e.Cancel = False
            tagGrid.Rows(e.RowIndex).ErrorText = ""
        Else
            e.Cancel = True
            MsgBox(tagDesc & " must match pattern:  " & regexDesc, MsgBoxStyle.Critical)
        End If
    Else
        e.Cancel = False
        tagGrid.Rows(e.RowIndex).ErrorText = ""
    End If
End Sub
    bool _fExceptionIsFatal = false;
    private void From1_Closing ( object sender , FormClosingEventArgs e )
    {
        if (this.CausesValidation )
        {   // There is no sense repeating this procedure if another routine already did it.
            DisableValidation ( );
        }   // if (this.CausesValidation )          

        if ( _fExceptionIsFatal )
        {   // Cancel False == Allow form to close.
            e.Cancel = false;
        }   // if ( _fExceptionIsFatal )
    }   // From1_Closing
    private void DisableValidation ( )
    {
        foreach ( Control ctrl in this.Controls )
        {
            ctrl.CausesValidation = false;
        }   // foreach ( Control ctrl in this.Controls )

        this.CausesValidation = false;
    }   // DisableValidation
protected override void WndProc(ref Message m)
{
    if (m.Msg == 0x10) // The upper right "X" was clicked
    {
        this.ActiveControl = null;
        this.AutoValidate = AutoValidate.Disable;
    }
    base.WndProc(ref m);
 }
MyForm.CausesValidation = false;
MyForm.AutoValidate = AutoValidate.EnableAllowFocusChange;
this.FormClosing += Form1_FormClosing;
private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
    if (this.CausesValidation)
    {
        DisableValidation();    
        this.Close();
    }
}
private void DisableValidation()
{
    txtbox1.CausesValidation = false;
    txtbox2.CausesValidation = false;
    CausesValidation = false;
}