C# 如何使用WinForms数据绑定正确触发UserControl中值的更改?

C# 如何使用WinForms数据绑定正确触发UserControl中值的更改?,c#,winforms,data-binding,user-controls,C#,Winforms,Data Binding,User Controls,我在WinForms应用程序中创建了两个UserControls。一个包含一个TextBox(现在我们称之为TextEntryControl),另一个应该使用我在TextBox中输入的值来执行内部操作(启用一个按钮并在单击此按钮时使用该值)-让我们称之为TextUsingControl 然而,我并没有通过数据绑定来实现这一点 第一(幼稚)方法 我向TextEntryControl添加了一个string属性,如下所示: public string MyStringProperty { get; s

我在WinForms应用程序中创建了两个
UserControl
s。一个包含一个
TextBox
(现在我们称之为
TextEntryControl
),另一个应该使用我在
TextBox
中输入的值来执行内部操作(启用一个按钮并在单击此按钮时使用该值)-让我们称之为
TextUsingControl

然而,我并没有通过数据绑定来实现这一点

第一(幼稚)方法

我向
TextEntryControl
添加了一个
string
属性,如下所示:

public string MyStringProperty { get; set; }
public string MyStringProperty {
    get {
        return _stringWrapper.Content;
    }
    set {
        _stringWrapper.Content = value;
    }
}
然后,我使用UI设计器将
TextBox
Text
属性绑定到此
MyStringProperty
,从而生成一个
textEntryControlBindingSource
。我在
InitializeComponents()后面添加了构造函数:

我向
TextUsingControl
添加了相同的属性,在我使用这两个控件的外部UI中,我将
TextUsingControl
的string属性绑定到
texentrycontrol
中的一个,并相应地更新了绑定源:

textEntryControlBindingSource.Add(textEntryControl1);
我在
TabControl
的不同选项卡上使用这两个控件,当我首先在文本框中输入文本,然后切换到另一个控件时,该机制只工作一次

下一次尝试

我为字符串创建了一个简单的包装器类:

public sealed class StringWrapper {

    public string Content { get; set; }
}
在我的文本输入控件中,我将文本框绑定到此字符串包装器,并将属性更改为如下所示:

public string MyStringProperty { get; set; }
public string MyStringProperty {
    get {
        return _stringWrapper.Content;
    }
    set {
        _stringWrapper.Content = value;
    }
}
我在外部控件中使用
TabControl
-使用
StringWrapper
将两个用户控件的
MyStringProperty
绑定到

结果:相同。但这是合乎逻辑的,因为委托给包装器的外部属性不会得到通知

第三次尝试

这是一个可行的办法,但我认为这是一个丑陋的解决办法

我完全抛弃了
MyStringProperty
,通过一个属性传入包装器对象本身,该属性再次将其传递给绑定源:

public StringWrapper MyStringWrapper {
    get {
        return stringWrapperBindingSource.Cast<StringWrapper>().FirstOrDefault();
    }
    set {
        stringWrapperBindingSource.Clear();
        if(value != null) stringWrapperBindingSource.Add(value);
    }
}
public StringWrapper MyStringWrapper{
得到{
返回stringWrapperBindingSource.Cast().FirstOrDefault();
}
设置{
stringWrapperBindingSource.Clear();
if(value!=null)stringWrapperBindingSource.Add(value);
}
}
现在,我只创建一个
StringWrapper
对象,并在
InitializeComponent()
之后将其设置为两个用户控件

INotifyPropertyChanged

作为后续行动:我尝试了
INotifyPropertyChanged
,并且描述了这也没有帮助

我想要实现的目标

我希望两个用户控件都有一个
MyStringProperty
,当我在
TextEntryControl
的文本框中输入的文本发生更改时,该属性应更新并正确通知它所附加到的任何绑定源。
TextUsingControl
应在其属性更改时更新自身

第二部分很简单,我只是在属性的
集合中添加了适当的逻辑,但是第一部分遇到了问题


我已经习惯了Eclipse的JFace数据绑定,在这里,这个功能可以通过
PropertyChangeSupport
PropertyChangeListener
来实现——在这里,我只需将适当的事件触发代码添加到setter中,就可以使用
BeanProperties.value()
设置数据绑定时。

它是正确的属性实现和正确的数据绑定的组合

(1)属性实现:

属性不需要复杂。它可以是简单的类型,就像您天真的方法一样,但关键是它应该提供属性更改通知。您可以使用常规
INotifyPropertyChanged
机制或Windows窗体特定的
PropertyNameChanged
命名事件模式。在这两种情况下,您都不能使用C#自动属性功能,而必须手动实现它(使用显式支持字段)。下面是一个示例实现:

string myStringProperty;
public string MyStringProperty
{
    get { return myStringProperty; }
    set
    {
        if (myStringProperty == value) return;
        myStringProperty = value;
        var handler = MyStringPropertyChanged;
        if (handler != null) handler(this, EventArgs.Empty);
    }
}

public event EventHandler MyStringPropertyChanged;
(2)数据绑定:

将单个控件属性绑定到单个对象属性称为简单数据绑定,通过以下方式实现。您可以查看方法/重载和类属性/方法/事件以了解更多信息

绑定所做的基本上是在源对象属性和目标对象属性之间创建链接(单向或双向)。请注意单词property-这正是开箱即用支持的内容。但使用以下简单的帮助器方法(在我的答案中介绍)可以轻松创建单向表达式,如绑定:

public static void Bind(this Control target, string targetProperty, object source, string sourceProperty, Func<object, object> expression)
{
    var binding = new Binding(targetProperty, source, sourceProperty, true, DataSourceUpdateMode.Never);
    binding.Format += (sender, e) => e.Value = expression(e.Value);
    target.DataBindings.Add(binding);
}
在演示中可以看到的另一个有趣的点是,如果您愿意,也可以实际绑定内部控制属性。
TextEntryControl
内部文本框和属性之间的整个同步是通过这一行实现的:

textConsumer.DataBindings.Add("MyStringProperty", textEntry, "MyStringProperty", true, DataSourceUpdateMode.Never);
textBox.DataBindings.Add("Text", this, "MyStringProperty", true, DataSourceUpdateMode.OnPropertyChanged);
TextConsumingControl
内部按钮启用:

button.Bind("Enabled", this, "MyStringProperty", value => !string.IsNullOrEmpty(value as string));
当然,最后两件事是可选的,您可以在属性setter中执行,但是知道这样的选项存在是很酷的