要使用验证处理程序,例如下面的“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();
}