所以,我在WinForms中遇到了这个问题,至少我认为这是一个问题,我无法弄清楚如何解决它。
基本上,当我为控件实现Validating
事件处理程序时,当控件未通过验证时,我将CancelEventArgs.Cancel
设置为true。示例代码:
private void NameTextBox_Validating(object sender, CancelEventArgs e) {
// Assume Cool Validation Logic.
//
// ...
var isValid = false;
if (!isValid) {
e.Cancel = true;
// Set an Error Provider Message.
//
// ...
}
}
这导致整个UI挂起。我无法选择任何其他控件,最小化父窗体窗口,或事件退出父窗体窗口。所以我找到了一个类似的问题,其答案建议为父表单实现FormClosing
事件处理程序,并在那里将FormClosingEventArgs.Cancel
设置为false。
问题是,只要我的控件无效,就不会调用该事件处理程序。我觉得它可能与我的控制层次结构有关,尽管我不是100%肯定。如果这有帮助,那就是:
-- Parent Form
---- Custom User Control
------ Text Box Control (<-- Causes Hang)
任何想法都会受到赞赏。
答案 0 :(得分:6)
我找到了一个如此简单的解决方案,实际上我感到很沮丧,因此我浪费了很多时间将我的头撞在键盘上。
只需将父容器的ContainerControl.AutoValidate
属性设置为EnableAllowFocusChange
,问题就解决了。在此上下文中,“父容器”表示实际容纳输入控件的容器。
如果您有控件层次结构,还可以将ContainerControl.AutoValidate
属性设置为Inherit
,只需将最外面的容器设置为EnableAllowFocusChange
。
希望这有助于将来。
答案 1 :(得分:-1)
您可以使用async / await或BackgroundWorker来处理UI线程的验证:
// in your init code...
textbox.TextChanged+=(s,e)=>Validate();
// and this is the Validation worker
BackgroundWorker validateWorker = null;
bool isValid = false;
private void Validate(){
if (validateWorker != null){
validateWorker.CancelAsync();
}
isValid = false;
string validateValue = textbox.Text;
BackgroundWorker localCopy = validateWorker = new BackgroundWorker();
validateWorker.WorkerSupportsCancellation = true;
validateWorker.DoWork+=(s,e)=>{
if (CoolValidationAssumed(validateValue) && !localCopy.CancellationPending)
isValid = true;
};
validateWorker.RunWorkerCompleted += (s, e) =>{
if (!localCopy.CancellationPending && !isValid)
textbox.Color = Colors.Red;
}
validationWorker.RunWorkerAsync();
}