3

単純な wpf アプリケーションを作成していますが、立ち往生しています。私はフィルタークラスを持っていることを達成したいと思います.idがユーザー入力によってフィルタークラスで変更された場合、フィルターを適用してリストを更新する必要があります。すべての初期バインディングが機能しています。リストは CompanyId と共に適切に表示されます。

xaml のデータバインディング:

<ListBox Height="212" HorizontalAlignment="Left" Margin="211,31,0,0" Name="listBoxProducts" VerticalAlignment="Top" Width="267" ItemsSource="{Binding ElementName=this, Path=Products}" DisplayMemberPath="CompanyId" />  
<TextBox Height="28" HorizontalAlignment="Left" Margin="12,31,0,0" Name="textBoxCompanyId" VerticalAlignment="Top" Width="170" Text="{Binding ElementName=this, Path=Company.Id}" />

xaml の分離コード:

private TestFactory _testFactory  = new TestFactory();

    private Company _company;
    public Company Company
    {
        get { return _company; }
    }

    private IProductList _products;
    public IProductList Products
    {
        get { return _products; }
    }


    public MainWindow()
    {
        _company = _testFactory.Company;
        _products = _testFactory.Products;

        InitializeComponent();
        _company.FilterChanged += _testFactory.FilterChanging;
    }

(ダミー)ファクトリークラス:

private IProductList _products;
    public IProductList Products 
    {
        get { return _products; }
    }

    private Company _company = new Company();
    public Company Company
    {
        get { return _company; }
    }

    public TestFactory()
    {
        _company = new Company() { Id = 2, Name = "Test Company" };
        GetProducts();
    }

    public void GetProducts()
    {
        var products = new List<Product>();
        products.Add(new Product() { ProductNumber = 1, CompanyId = 1, Name = "test product 1" });
        products.Add(new Product() { ProductNumber = 2, CompanyId = 1, Name = "test product 2" });
        products.Add(new Product() { ProductNumber = 3, CompanyId = 2, Name = "test product 3" });

        if (Company.Id != 2)
        {
            products = products.Where(p => p.CompanyId == Company.Id).ToList();
        }

        _products = new ProductList(products);
    }

    public void FilterChanging(object sender, EventArgs e)
    {
        GetProducts();
    }

ProductList インターフェース:

public interface IProductList : IList<Product>, INotifyCollectionChanged {}

製品リスト クラス:

public class ProductList : IProductList
{
    private readonly IList<Product> _products;

    public ProductList() { }

    public ProductList(IList<Product> products)
    {
        _products = products;
    }


    public IEnumerator<Product> GetEnumerator()
    {
        return _products.GetEnumerator();
    }


    IEnumerator IEnumerable.GetEnumerator()
    {
        return GetEnumerator();
    }


    public void Add(Product item)
    {
        _products.Add(item);
        notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item));
    }


    public void Clear()
    {
        _products.Clear();
        notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
    }


    public bool Contains(Product item)
    {
        return _products.Contains(item);
    }


    public void CopyTo(Product[] array, int arrayIndex)
    {
        _products.CopyTo(array, arrayIndex);
    }


    public bool Remove(Product item)
    {
        var removed = _products.Remove(item);

        if (removed)
        {
            notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item));
        }
        return removed;
    }


    public int Count
    {
        get { return _products.Count; }
    }


    public bool IsReadOnly
    {
        get { return _products.IsReadOnly; }
    }


    public int IndexOf(Product item)
    {
        return _products.IndexOf(item);
    }


    public void Insert(int index, Product item)
    {
        _products.Insert(index, item);
        notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
    }


    public void RemoveAt(int index)
    {
        _products.RemoveAt(index);
        notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
    }


    public Product this[int index]
    {
        get { return _products[index]; }
        set
        {
            _products[index] = value;
            notifyCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace, _products[index]));
        }
    }


    public event NotifyCollectionChangedEventHandler CollectionChanged;
    private void notifyCollectionChanged(NotifyCollectionChangedEventArgs args)
    {
        if (CollectionChanged != null)
        {
            CollectionChanged(this, args);
        }
    }
}

Company クラス (フィルター クラス):

public class Company : INotifyPropertyChanged
{
    private int _id;
    public int Id
    {
        get { return _id; }
        set 
        {
            if (_id == value)
                return;

            _id = value;
            OnPropertyChanged("Id");

            OnFilterChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
        }
    }

    private string _name;
    public string Name
    {
        get { return _name; }
        set
        {
            if (_name == value)
                return;

            _name = value;
            OnPropertyChanged("Name");
        }
    }


    public event PropertyChangedEventHandler PropertyChanged;
    public event EventHandler FilterChanged;

    private void OnPropertyChanged(string name)
    {
        if (PropertyChanged == null)
            return;

        var eventArgs = new PropertyChangedEventArgs(name);
        PropertyChanged(this, eventArgs);
    }

    private void OnFilterChanged(NotifyCollectionChangedEventArgs e)
    {
        if (FilterChanged == null)
            return;

        FilterChanged(this, e);
    }
}

リストは工場出荷時に更新されますが、ビューに変更はありません。私はおそらく何か間違ったことをしています。私のアプローチ全体が最善ではないのかもしれません。たぶん、valueconverter で ObservableCollection タイプを使用する必要がありますか? どんな助けでも大歓迎です。乾杯!

4

2 に答える 2

5

ObservableCollection<Product>に基づいて独自のリストを作成する代わりに、を使用します。IList

の目的はObservableCollectionコレクションへの変更を追跡することであり、コレクションが変更されると自動的に UI を更新します。

于 2012-01-26T16:18:02.427 に答える
0

このユース ケースでは、ICollectionView の使用も検討してください。

この投稿を参照してください。

于 2012-01-26T16:26:57.907 に答える