Лучший способ заставить привязанный к данным WPF ListBox для обновления? - PullRequest
29 голосов
/ 31 октября 2008

У меня есть WPF ListBox, который связан с ObservableCollection, при изменении коллекции все элементы обновляют свои позиции.

Новая позиция сохраняется в коллекции, но пользовательский интерфейс не обновляется. Поэтому я добавил следующее:

    void scenarioItems_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
    {
        ToolboxListItem.UpdatePositions();
        lstScenario.ItemsSource = null;
        lstScenario.ItemsSource = ToolboxListItem.ScenarioItems;
        this.lstScenario.SelectedIndex = e.NewStartingIndex;
    }

Если для ItemsSource задать значение null, а затем снова связать его, пользовательский интерфейс будет обновлен,

но это, вероятно, очень плохое кодирование: p

Предложения

Ответы [ 7 ]

74 голосов
/ 31 октября 2008

У меня есть список, связанный со свойством объекта типа List<MyCustomType>(), и я убедился, что следующий код обновляет список при обновлении списка.

void On_MyObjProperty_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
{
   MyListBox.Items.Refresh();
}

Если проблемы все еще возникают, отсканируйте окно вывода VS IDE (Ctrl + W, O) и посмотрите, сможете ли вы обнаружить какие-либо сообщения об ошибках привязки.

10 голосов
/ 05 декабря 2009

WPF привязывает список / коллекцию элементов к ListBox, но пользовательский интерфейс не обновляется после обновления элементов, Решено .

Я просто тупой. Хотя я много читал об использовании ObservableCollection<> вместо List<>, я просто продолжал игнорировать это предложение и следовал другим советам, но безрезультатно. Вернулся к моим книгам и перечитал. Довольно хорошо объяснено, что ObservableCollection<> является обязательным для использования, потому что List<> не предоставляет интерфейс INotifyCollectionChange, необходимый для ListBox для обновления отображения при изменении элементов в коллекции.

Это обновленный код:

private ObservableCollection<StringWrapper> m_AppLog;
ObservableCollection<StringWrapper> Log { get { return m_AppLog; } }

Довольно просто и не требует ничего другого (например, Refresh ()). Поскольку ObservableCollection заботится о срабатывании события изменения, я смог удалить ненужный вызов:

// notify bound objects
OnPropertyChanged("Log");

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

public void AddToLog(string message) {
    if (Thread.CurrentThread != Dispatcher.Thread) {
        // Need for invoke if called from a different thread
        Dispatcher.Invoke(
            DispatcherPriority.Normal, (ThreadStart)delegate() { AddToLog(message); });
    }
    else {
        // add this line at the top of the log
        m_AppLog.Insert(0, new StringWrapper(message));
        // ...

Также обратите внимание, что ObservableCollection<> не поддерживает RemoveRange() вопреки List<>. Это часть возможных корректировок, требуемых при переходе от List к ObservableCollection.

6 голосов
/ 10 ноября 2010

У меня может быть та же проблема, что и у вас, но я не уверен.

У меня были ObservableCollection<MyEntity> и ListBox привязанные к нему. Но по какой-то странной причине мой ListBox не обновлялся, когда я изменил свойства MyEntity объектов в списке.

После поиска я нашел следующую страницу, и мне просто пришлось сообщить вам:

http://www.wblum.org/listbind/net3/index.html

Это очень хорошее описание того, что вам нужно сделать, чтобы ListBox обновлялся при изменении списка или объектов в нем. Надеюсь, что вы выиграете от этого.

4 голосов
/ 31 октября 2008

Вчера у меня была такая же проблема, и это полный кусок дерьма :) ... хотя я больше не устанавливаю свой ноль. В моем сценарии я устанавливаю его в MyList.ToArray () (после каждого добавления в список).

Я видел несколько слов "о, вам нужно использовать ObservableList" <- полная чушь. </p>

Я видел несколько «о, вызовите« Обновить »» <- полная чушь. </p>

Прошу простить меня, но я также ожидал, что это сработает:)

2 голосов
/ 13 апреля 2010

Это старые вещи, но используйте ObservableCollection. Если вы хотите, чтобы пользовательский интерфейс видел обновления свойств в объектах ObservableCollection, вам необходимо реализовать INotifyPropertyChanged в определении класса для этого объекта. Затем создайте событие свойства изменено в установщике каждого свойства.

Public Class Session
Implements INotifyPropertyChanged

Public Event PropertyChanged As PropertyChangedEventHandler _
   Implements INotifyPropertyChanged.PropertyChanged

Private Sub NotifyPropertyChanged(ByVal info As String)
    RaiseEvent PropertyChanged(Me, New PropertyChangedEventArgs(info))
End Sub

Private _name As String = "No name"
''' <summary>
''' Name of Session
''' </summary>
''' <value></value>
''' <returns></returns>
''' <remarks></remarks>
Public Property Name() As String
    Get
        Return _name
    End Get
    Set(ByVal value As String)
        _name = value
        NotifyPropertyChanged("Name")
    End Set
End Property
0 голосов
/ 10 октября 2014

Для меня это больше похоже на ошибку в ListBox и ListView. Я привязан к ObservableCollection, элементы в коллекции реализуют INotifyPropertyChanged. Пользовательский интерфейс не показывает добавленные элементы, когда я динамически нажимаю кнопку «Добавить элемент», однако у меня есть счетчик, связанный с MyCollection.Count. Это управление счетчиком увеличивается каждый раз, когда я нажимаю кнопку «Добавить элемент». Если я изменю размер представления, в окне списка отображаются все мои добавленные элементы. Таким образом, привязка ItemSource к элементу управления ListBox нарушена. Я также позаботился о том, чтобы в любой момент не создавать новую коллекцию MyCollection, которая бы нарушила привязку. Бу Ху.

0 голосов
/ 26 апреля 2009

Если у вас есть ObservableList объектов, и вы изменяете свойства внутри этих объектов, уведомление не применяется, так как коллекция не изменяется напрямую. Я заставляю уведомление после изменения свойств моего объекта, используя Insert (), чтобы повторно добавить мой измененный объект в коллекцию, затем RemoveAt (), чтобы удалить старую копию. Это не красиво, но работает.

...