3

我有一个 ObservableCollection 并希望将一个文本框绑定到该集合的特定元素。ObservableCollection 中的项目属于实现 INotifyPropertyChanged 的​​类型。

我曾考虑过创建一个从 ObservableCollection 中选择正确元素的属性,但是当集合中的相应元素发生更改时,我必须让该属性实现,我不确定这是否是正确的方法。

4

3 回答 3

5

通常,特别是如果您使用 MVVM,您将拥有一个带有 ObservableCollection 的 viewModel 和一个使用数据绑定更新的 SelectedItem 属性。

例如,您的 viewModel 可能如下所示:

    class ProductsViewModel : INotifyPropertyChanged
{
    public ObservableCollection<Product> Products { get; set; }
    private Product _selectedProduct;

    public Product SelectedProduct
    {
        get { return _selectedProduct; }
        set 
        { 
            _selectedProduct = value; 
            if (PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs("SelectedProduct"));
        }
    }



    public ProductsViewModel()
    {
        Products = new ObservableCollection<Product>();
        Products.Add(new Product() { Name = "ProductA" });
        Products.Add(new Product() { Name = "ProductB" });
    }

    public event PropertyChangedEventHandler PropertyChanged;
}

您的窗口对象 xaml:

<Window x:Class="ProductsExample.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525">
    <Grid>
        <ListBox HorizontalAlignment="Left" Height="171" Margin="32,29,0,0" VerticalAlignment="Top" Width="176"
                 ItemsSource="{Binding Products}"
                 SelectedItem="{Binding SelectedProduct, Mode=TwoWay}"
                 DisplayMemberPath="Name"
                 />
        <TextBox HorizontalAlignment="Left" Height="33" Margin="36,226,0,0" TextWrapping="Wrap" VerticalAlignment="Top" Width="172"
                 Text="{Binding SelectedProduct.Name, Mode=TwoWay}"/>

    </Grid>
</Window>

以及您刚刚设置数据上下文的代码隐藏:

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        this.DataContext = new ProductsViewModel();
    }
}

每当您在列表框中选择产品时,文本框都会使用所选产品进行更新,如果您更改文本框中的产品(如果产品正确实现了 INotifyPropertyChanged),则列表框中的项目也将更新。

显然,您可以仅使用代码隐藏来实现所有这些,但这里解释了几个原因:http: //msdn.microsoft.com/en-us/magazine/dd419663.aspx,最好有一个 ViewModel

于 2013-08-28T22:40:05.343 回答
4

如果您需要的项目是特定于索引的,您可以使用索引访问

 <TextBlock Text="{Binding MyItemsSource[2]}" />
于 2013-08-28T23:01:50.230 回答
3

为了解决我的问题,我创建了一个从 ObservableCollection 中选择正确元素的属性,并创建了一个事件处理程序,该处理程序被添加到 ObservableCollection 的 CollectionChanged 事件中,并为我的 SelectionProperty 引发 PropertyChanged 事件。

在包含 ObservableCollection 和 SelectionProperty 的类的构造函数中的代码看起来像这样:

myObservableColleciton.CollectionChanged += 
        new NotifyCollectionChangedEventHandler(
        myObservableCollection_CollectionChanged);

类中的其他地方定义此事件处理程序:

void myObservableCollection_CollectionChanged(
        Object sender, NotifyCollectionChangedEventArgs e){
   if (PropertyChanged != null)
   {
       PropertyChanged(this, new PropertyChangedEventArgs("SelectionProperty"));
   }
}

我的 selectionProperty 看起来像这样:

public User SelectionProperty
{
     get { return myObservableCollection.First( user => user.id == 0); }
}

如果 SelectionProperty 依赖的不仅仅是 ObservableCollection(也许我们想找到一个最接近某个年龄的用户,这是在其他地方设置的),那么需要确保 SelectionProperty 的 PropertyChanged 事件也被引发,当那些其他属性改变。

于 2013-09-02T13:00:09.077 回答