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