我有一个绑定到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时调用。
谢谢!
答案 0 :(得分:0)
这就是绑定的工作方式。除非目标通过绑定本身以外的方式更改,否则不会更新源或源。即它假设如果目标刚刚从源更新,那么源(已)是最新的,不需要更新。
如果没有更多细节,很难确切知道自己想要什么。但似乎您可能要求NewName
实际上是第二个绑定的目标,其中源是相同的Name
属性,用作TextBox.Text
属性,或者您希望订阅TextBox.TextChanged
事件,并且您的处理程序在引发该事件时显式地将值写回NewName
属性。
在前一种情况下,您必须使NewName
成为MainWindow
的依赖属性。这可能是您可能想要或可能不想处理的并发症。如果没有,那么我建议采用后一种方法。