好的,所以这里是我的问题的一些上下文,写成伪C#代码(随意指出任何错误):(你可以直接跳转到堆栈跟踪并稍后阅读上下文。)
public class SomeForm {
private _model = new ViewModelClass
public void new() {
// Normal Winforms init omitted
ViewModelClassBindingSource.DataSource = _model;
SomeControl1.SetModel(_model);
}
}
public class SomeControl {
private _model = new ViewModelClass
internal void SetModel(ViewModelClass model) {
_model = model;
ViewModelClassBindingSource.DataSource = model;
ViewModelClassBindingSource.ResetBindings(true);
}
}
public class ComplexObject : IPropertyChanging, IPropertyChanged {
public property bool BoolProp {get; set;}
}
public class ViewModelClass : IPropertyChanged {
property IList<ComplexObject> ComplexObjects {get;}
property ComplexObject SelectedComplexObject {get; set;}
property Object SomethingNotNecessarilyRelated {get; set;}
private void NotifyPropertyChanged(string propName) {
PropertyChanged(this, new PropertyChangedEventArgs(propName));
}
}
这些类中提到的所有属性都是在Visual Studio 2008 Windows窗体设计器,SomeForm
或SomeControl
类中的数据绑定。 (ComplexObject.BoolProp
两者都是数据绑定)。不要犹豫,就上下文提出更多问题。
问题:当我在ViewModelClass
类中制作一些(一堆)通知时,有一种下意识的反应将ComplexObject.BoolProp
设置为false
,使用这种堆栈跟踪:
System.dll!System.ComponentModel.ReflectPropertyDescriptor.SetValue(object component = "Object Exposed in 'SelectedComplexObject'", object value = false) + 0x124 bytes
System.Windows.Forms.dll!System.Windows.Forms.BindToObject.SetValue(object value) + 0x5d bytes
System.Windows.Forms.dll!System.Windows.Forms.Binding.PullData(bool reformat, bool force) + 0x15a bytes
System.Windows.Forms.dll!System.Windows.Forms.BindingManagerBase.PullData(out bool success = true) + 0x6e bytes
System.Windows.Forms.dll!System.Windows.Forms.BindingSource.ParentCurrencyManager_CurrentItemChanged(object sender = {System.Windows.Forms.CurrencyManager}, System.EventArgs e) + 0x54 bytes
[Native to Managed Transition]
[Managed to Native Transition]
System.Windows.Forms.dll!System.Windows.Forms.CurrencyManager.OnCurrentItemChanged(System.EventArgs e) + 0x17 bytes
System.Windows.Forms.dll!System.Windows.Forms.CurrencyManager.List_ListChanged(object sender, System.ComponentModel.ListChangedEventArgs e) + 0x3bc bytes
System.Windows.Forms.dll!System.Windows.Forms.BindingSource.OnListChanged(System.ComponentModel.ListChangedEventArgs e) + 0x7e bytes
System.Windows.Forms.dll!System.Windows.Forms.BindingSource.InnerList_ListChanged(object sender, System.ComponentModel.ListChangedEventArgs e) + 0x2e bytes
System.dll!System.ComponentModel.BindingList<System.__Canon>.OnListChanged(System.ComponentModel.ListChangedEventArgs e) + 0x17 bytes
System.dll!System.ComponentModel.BindingList<MyCompany.ViewModelClass>.Child_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e) + 0x176 bytes
[Native to Managed Transition]
[Managed to Native Transition]
MyCompany.Gui.exe!MyCompany.ViewModelClass.NotifyPropertyChanged(String propertyName = "SomethingNotNecessarilyRelated") Line 437 + 0x3c bytes Basic
为什么该程序想将SomeBool
设置为false
?我怎么能防止这种情况?
关于Stack Overflow的第一个问题是关于Windows Forms应用程序中包含意外值的字段,就像你的一样。解决方案是等到表单加载事件被触发以设置表单的GUI元素。
我会推迟设置_model
(包括用new
构建它)和其他GUI元素,直到在表单加载事件的处理程序中。
如何:
在Visual Studio中添加表单加载处理程序:
SomeForm.cs
)SomeForm_Load
的函数添加框架代码,并且行this.Load += new System.EventHandler(this.SomeForm_Load);
将添加到SomeForm.Designer.cs
。将设置代码移动到SomeForm_Load
:
private void SomeForm_Load(object aSender, EventArgs anEvent)
{
_model = new ViewModelClass;
ViewModelClassBindingSource.DataSource = _model;
SomeControl1.SetModel(_model);
}
从= new ViewModelClass
的声明中删除“_model
”。
我已将单个ViewModelClass
重构为SomeFormViewModel
和SomeControlViewModel
,并将它们绑定到各自的类别。
简单地这样做就会使问题消失。
我仍然希望更好地理解我所放置的堆栈跟踪 - 我猜想问题的关键是每个BindingSource
都保留自己对对象所做更改的信息 - 并且因为它们都对它们进行了更改对象,他们不知道发生了什么事。
有同样的问题,在tabcontrol中更改选项卡会将我的所有数据绑定布尔值重置为“false”。所有非布尔值都很好。堆栈跟踪显示与OP完全相同。
试图像Peter建议的那样将视图模型的设置移动到Form_Load,但没有运气。通过创建数据源并在所有控件属性上设置数据绑定,最终放弃并将所有数据绑定从代码移动到在UI中设置它。