14

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

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

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

但是,当我输入不正确的值并关闭表单(使用右上角的关闭按钮)时,它会验证文本框内容并抛出消息框,并且在我取消事件时表单不会关闭。

问题是,当我单击表单右上角的 X 按钮时,我不希望触发验证,因为无论如何我都要关闭表单。我怎样才能做到这一点?

我会尽快发布代码片段。

4

10 回答 10

19

要使用验证处理程序,例如下面的“txtIPAddress_Validating()”处理程序,同时能够关闭表单而无需输入有效条目,我执行以下操作:

1) 启动验证处理程序:从您希望为其激活验证的控件的控件属性中,双击此控件事件列表中的“验证”事件。通过单击此控件的属性表的事件(闪电般的)工具栏按钮可以访问控件事件列表。然后,您可以在自动生成的处理程序中输入代码,其名称结合了控件名称和“_Validating”。该处理程序中将某些内容确定为无效的部分可以通过添加“e.Cancel = true”指令来强制输入有效条目。

对于此类验证方法示例,请参阅下面的“txtIPAddress_Validating()”和“txtMaskBits_Validating()”代码。不要被这些特定示例的完整验证机制分心。为了强制验证,您需要在自己的代码中查看和重现,只需在您自己的验证方法的正确位置添加“e.Cancel = true”指令。即当值被识别为无效时。

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

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

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

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

c) 要使此“取消”按钮正常工作,您还需要将此“取消”按钮的“CausesValidation”属性设置为“假”。这是必要的,否则单击此按钮将触发验证,然后可以通过上面的“AutoValidate = AutoValidate.Disable;”禁用验证 操作说明。

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

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

这里的挑战是在执行验证之前捕获此类“X”单击事件。任何通过表单关闭处理程序执行此操作的尝试都将不起作用,因为一旦执行到达此类处理程序就为时已晚。但是,可以通过覆盖 WndProc() 方法并测试“m.Msg == 0x10”条件来迅速捕获“X”按钮的单击。当该条件为真时,前面介绍的“AutoValidate = AutoValidate.Disable;” 在这种情况下,指令也可以再次用于禁用整体验证。有关此类方法的代码示例,请参见下面的 WndProc() 方法。您应该能够在表单类中复制和粘贴该方法。

此时,“取消”和“X”按钮都应该取消验证。但是,可用于关闭表单的转义键却没有。当使用表单的“CancelButton”属性将此转义键链接到表单的“取消”按钮时,会激活此类转义键。

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();
    }
于 2013-10-01T14:32:25.677 回答
4

在文本框的验证事件中插入以下内容作为第一行:

//Allow the form to be closed
if (this.ActiveControl.Equals(sender))
    return;

由于表单的关闭事件正在触发验证,并且(通常至少)是唯一会触发验证的表单事件,我们可以假设任何触发验证的表单事件都是关闭事件。

于 2014-03-09T02:21:59.270 回答
3

与这里的所有建议相比,实际答案非常简单,这些建议涉及黑客和额外多余的代码来撤消某些东西。

“技巧”只是让焦点改变而不是从表单本身的按钮触发验证。

您可以简单地在表单上设置两个属性:

MyForm.CausesValidation = false;
MyForm.AutoValidate = AutoValidate.EnableAllowFocusChange;

瞧,当您尝试关闭它时,表单行为正常,并且在其他输入(例如tab更改焦点或鼠标单击)后验证仍然有效。

于 2016-11-28T09:39:18.410 回答
2
private void Form1_FormClosing(object sender, FormClosingEventArgs e) {
    // Assume that X has been clicked and act accordingly.
}

创建一个结束事件,然后简单地取消您的验证器。

于 2013-04-10T08:36:54.273 回答
1

尝试设置CauseValidation为假

或在这里查看:单击表单的取消按钮后如何跳过验证

或者尝试在 formClosing 事件中设置它

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

}
于 2013-04-10T08:48:19.120 回答
1

您需要的是一个类似于下面描述的实现,假设您有一个保存按钮和一个取消按钮的表单:

    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();
    }
于 2014-03-12T04:33:39.657 回答
0

在验证检查中检查哪个按钮具有焦点。如果取消按钮(在我的情况下是清除按钮),请覆盖。这是我从单元格验证事件处理程序调用的内部方法。(刚刚意识到这是一个 C# 问题,但你可以翻译)

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
于 2014-11-25T14:02:09.033 回答
0

我来这里是为了寻找一种简单的方法来在 Validating 事件处理程序引发异常、报告异常并需要强制表单关闭时导致表单关闭。在阅读了这个主题和许多其他主题之后,经过一个下午的实验,我有了几个发现,并开发了一个简单的 hack 来强制关闭表单。

不过,首先要做的是;我发现当验证事件调用this.Close()时,传递到其From_Closing事件过程的FormClosingEventArgs.Cancel标志设置为TRUE,从而有效地导致事件自行取消。相反,正常的 Close 事件会收到设置为FALSE的FormClosingEventArgs.Cancel标志。

由于表单上的Close方法不带参数,因此没有直接的方法来强制执行此问题,从而需要进行 hack。本文讨论了一些这样的 hack,但我认为我的实现起来要简单得多。

hack 从一个简单的表单级布尔变量开始。

    bool _fExceptionIsFatal = false;

除了定义Form_Closing事件处理程序之外,这是表单所需的唯一结构更改。

Form_Closing事件很简单。

    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

尽管DisableValidation是作为当前表单的本地方法实现的,但同样的事情可以通过将Form引用传递到库例程中来完成,因为 Form 是 Form,并且它的 Controls 集合是 Controls 集合。不久,我将这样做,同时实现它的逆,再次打开验证。

    private void DisableValidation ( )
    {
        foreach ( Control ctrl in this.Controls )
        {
            ctrl.CausesValidation = false;
        }   // foreach ( Control ctrl in this.Controls )

        this.CausesValidation = false;
    }   // DisableValidation

解决方案的第四部分同样简单;每当您想强制关闭表单时,请在调用this.Close之前将_intValueAsInteger设置为TRUE

于 2015-09-07T00:33:47.697 回答
0

将以下代码添加到您的表单中。即使子控件正在验证,您也可以关闭表单。

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);
 }
于 2015-10-18T02:07:22.493 回答
0

这个问题已经很老了,但无论如何都想添加答案:

在表单构造函数中:

this.FormClosing += Form1_FormClosing;

在关闭事件处理程序中(确保表单的 CausesValidation 设置为 true。您还可以检查文本框的 CausesValidation 属性而不是表单的):

private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
    if (this.CausesValidation)
    {
        DisableValidation();    
        this.Close();
    }
}

在 DisableValidation 方法中,禁用文本框和表单的验证(我使用的是 2):

private void DisableValidation()
{
    txtbox1.CausesValidation = false;
    txtbox2.CausesValidation = false;
    CausesValidation = false;
}
于 2020-02-05T05:50:29.930 回答