WPF: добавление элемента в коллекцию с привязкой к данным (свойство зависимости) - PullRequest
1 голос
/ 10 мая 2009

У меня есть DependencyProperty, который содержит сущность со свойством, являющимся коллекцией (ShoutBox.Entities):

public static readonly DependencyProperty ShoutBoxProperty = DependencyProperty.Register("ShoutBox",typeof (ShoutBox),typeof (ShoutBoxViewerControl));

public ShoutBox ShoutBox
{
    get { return (ShoutBox) GetValue(ShoutBoxProperty); }
    set { SetValue(ShoutBoxProperty, value); }
}

Он связывается в xaml, например:

<ItemsControl ItemsSource="{Binding ShoutBox.Entries}">
.
.
</ItemsControl>

Когда я связываю его в первый раз, он работает как положено, но бывают случаи, когда мне нужно добавлять элементы в коллекцию (с помощью метода, который находится в том же элементе управления), например:

public void AddNewEntry(ShoutBoxEntry newEntry)
{
    Dispatcher.Invoke(new Action(() =>{
        ShoutBox.Entries.Add(newEntry); //Adding directly the the Dependency property
    }));
}

Проблема в том, что когда я добавляю новый элемент описанным выше способом, элемент не отображается в ItemsControl.


У меня вопрос, , почему новый элемент, который я добавляю, не отображается в ItemsControl?


[Изменить]

Entries ( ShoutBox.Entries ) имеет тип List<ShoutBoxEntry>

Ответы [ 2 ]

3 голосов
/ 10 мая 2009

Какой тип записей? Он должен быть ObservableCollection или реализовывать ICollectionChanged. В противном случае привязка не будет знать, что был добавлен новый элемент.

0 голосов
/ 11 мая 2009

Изменение типа записей действительно должно решить проблему ... Если вы хотите избежать явного вызова Dispatcher.Invoke, я написал коллекцию, которая вызывает события CollectionChanged и PropertyChanged в потоке, создавшем коллекцию:

public class AsyncObservableCollection<T> : ObservableCollection<T>
{
    private SynchronizationContext _synchronizationContext = SynchronizationContext.Current;

    public AsyncObservableCollection()
    {
    }

    public AsyncObservableCollection(IEnumerable<T> list)
        : base(list)
    {
    }

    protected override void OnCollectionChanged(NotifyCollectionChangedEventArgs e)
    {
        if (SynchronizationContext.Current == _synchronizationContext)
        {
            // Execute the CollectionChanged event on the current thread
            RaiseCollectionChanged(e);
        }
        else
        {
            // Post the CollectionChanged event on the creator thread
            _synchronizationContext.Post(RaiseCollectionChanged, e);
        }
    }

    private void RaiseCollectionChanged(object param)
    {
        // We are in the creator thread, call the base implementation directly
        base.OnCollectionChanged((NotifyCollectionChangedEventArgs)param);
    }

    protected override void OnPropertyChanged(PropertyChangedEventArgs e)
    {
        if (SynchronizationContext.Current == _synchronizationContext)
        {
            // Execute the PropertyChanged event on the current thread
            RaisePropertyChanged(e);
        }
        else
        {
            // Post the PropertyChanged event on the creator thread
            _synchronizationContext.Post(RaisePropertyChanged, e);
        }
    }

    private void RaisePropertyChanged(object param)
    {
        // We are in the creator thread, call the base implementation directly
        base.OnPropertyChanged((PropertyChangedEventArgs)param);
    }
}

Более подробную информацию можно найти здесь: http://www.thomaslevesque.com/2009/04/17/wpf-binding-to-an-asynchronous-collection/

...