WPF Multibinding:从另一个绑定更新的TextBox的OneWayToSource绑定不起作用?

时间:2016-05-19 21:31:17

标签: c# wpf xaml multibinding imultivalueconverter

我有一个绑定到People集合的DataGrid。我还有一个TextBox,它应该接受所选行的Name值。然后,用户可以编辑该值,也可以保留原样。关键点是:TextBox中显示的文本无论是源自集合还是用户输入都必须传播到属性NewName。

我为NewNameTextBox设置了两个绑定:OneWay用于DataGrid后面的CollectionView,OneWayToSource用于属性:

<Window.Resources>
    <CollectionViewSource x:Key="PeopleCollection" 
         Source="{Binding Path=People, Mode=OneWay}" />
    <local:ConverterNewNamePrefill x:Key="ConverterNewNamePrefill" />
</Window.Resources>
<Grid>
    <StackPanel>
        <DataGrid ItemsSource="{Binding Source={StaticResource PeopleCollection}}"
                  AutoGenerateColumns="True"
                  IsReadOnly="True"
                  Margin="10">
        </DataGrid>
        <StackPanel Orientation="Horizontal" Margin="10">
            <TextBox>
                <TextBox.Text>
                    <MultiBinding Converter="{StaticResource ConverterNewNamePrefill}" >
                        <Binding Source="{StaticResource PeopleCollection}" Path="Name" Mode="OneWay" />
                        <Binding Path="NewName" Mode="OneWayToSource" />
                    </MultiBinding>
                </TextBox.Text>
            </TextBox>
        </StackPanel>
    </StackPanel>
</Grid>

我认为当用户更改DataGrid中的选择时,应该更新属性,但这不会发生。 TextBox会更新并显示所选的Name值,但通过OneWayToSource绑定的属性保持不变。

如果用户键入TextBox,则属性会按预期更新。

所以问题是如何在没有代码隐藏视图的情况下通过多边界TextBox更新两个源的属性?

以下是窗口背后的代码:

public class Person
{
        public string Name { get; set; }
        public string Surname { get; set; }
}

public partial class MainWindow : Window
{

    public MainWindow()
    {
        InitializeComponent();
    }

    private ObservableCollection<Person> _people = new ObservableCollection<Person> {
        new Person() {Name = "Mitchell", Surname = "Sofia" },
        new Person() {Name="Bush", Surname="Ethan" },
        new Person() {Name="Ferrero", Surname="Emma" },
        new Person() {Name="Thompson", Surname="Aiden" }
    };
    public ObservableCollection<Person> People => _people;

    public string NewName { get; set; } = "Jackson";
}

public class ConverterNewNamePrefill : IMultiValueConverter
{

    public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
    {
        return values[0]; 
    }

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, CultureInfo culture)
    {
        return new[] { value, value };
    }
}

转换器的方法ConvertBack()仅在用户输入时调用,而不是在从集合更新TextBox.Text时调用。

谢谢!

1 个答案:

答案 0 :(得分:0)

这就是绑定的工作方式。除非目标通过绑定本身以外的方式更改,否则不会更新源或源。即它假设如果目标刚刚从源更新,那么源(已)是最新的,不需要更新。

如果没有更多细节,很难确切知道自己想要什么。但似乎您可能要求NewName实际上是第二个绑定的目标,其中源是相同的Name属性,用作TextBox.Text属性,或者您希望订阅TextBox.TextChanged事件,并且您的处理程序在引发该事件时显式地将值写回NewName属性。

在前一种情况下,您必须使NewName成为MainWindow的依赖属性。这可能是您可能想要或可能不想处理的并发症。如果没有,那么我建议采用后一种方法。