我有一个绑定到ObservableCollection的WPF ListBox, 当集合发生变化时,所有项目都会更新其位置。

新位置存储在集合中,但UI不会更新。 所以我添加了以下内容:

    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然后再次绑定它,UI将被更新,

但这可能是非常糟糕的编码:p

建议?

有帮助吗?

解决方案

我有一个Listbox绑定到一个类型为List<MyCustomType>()的对象属性,并且我验证了以下代码在更新List时更新了列表框。

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

如果您仍然遇到问题,请扫描VS IDE输出窗口(Ctrl + W,O),看看是否可以发现报告的任何绑定错误。

其他提示

WPF将列表/项目集合绑定到ListBox,但更新项目后UI未刷新,已解决

我只是愚蠢。虽然我读了很多关于使用ObservableCollection<>而不是List<>的内容,但我只是继续忽略这个建议并继续关注其他建议,但无济于事。回到我的书并重读。很好地解释了INotifyCollectionChange是必须使用的,因为ListBox不提供ObservableCollection在集合中的项目更改时更新其显示所需的RemoveRange()接口。

这是更新的代码:

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

非常简单,不需要任何其他内容(例如Refresh())。因为ObservableCollection负责触发change事件,所以我能够删除不必要的调用:

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

<=>不支持未创建它的线程的更新。因为我的列表(显示最近的错误/信息消息的可视日志)可以从不同的线程更新,我添加以这种方式调整我的代码,以确保更新是使用列表自己的调度程序完成的:

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));
        // ...

另请注意,<=>不支持<=>与<=>相反。这是从List切换到ObservableCollection时所需的可能调整的一部分。

我可能遇到类似的问题,但我不确定。

我有一个ObservableCollection<MyEntity>ListBox绑定它。但是对于一些奇怪的原因当我更改列表中MyEntity对象的属性时,我的<=>没有被更新。

搜索了一段时间后,我找到了以下页面,我只是想让你知道:

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

这是一个非常好的描述,当列表或其中的对象发生更改时,您需要做什么才能更新<=>。希望你能从中受益。

昨天我遇到了同样的问题,这是一个完整的废话:) ...我不会再将我的设置为null了。在我的场景中,我将其设置为MyList.ToArray()(每次添加到列表后)。

我见过多个<!>“哦,你需要使用一个ObservableList <!>”; <!> lt; - 完全废话。

我见过多个<!>!哦,打电话'刷新'<!> <!> lt; - 完全废话。

请原谅我的沮丧,但我也希望这可行:)

这是旧的东西但是,使用ObservableCollection。如果您希望UI在ObservableCollection的Objects中查看属性的更新,则需要在该对象的类defenition中实现INotifyPropertyChanged。然后在每个属性的setter中引发属性更改事件。

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

如果您有Observable对象列表,并且您正在更改这些对象内的属性,则通知不会应用,因为集合不会直接更改。通过使用Insert()将我更改的对象重新添加到集合,然后使用RemoveAt()删除旧副本,我在更改对象属性后强制通知。它不漂亮,但它有效。

对我而言,它看起来更像是ListBox和ListView中的错误。我绑定了一个ObservableCollection,集合中的项目实现了INotifyPropertyChanged。当我动态按下“添加项目”按钮时,UI没有显示添加的项目,但是我有一个绑定到MyCollection.Count的计数器控件。每次按下“添加项目”按钮时,此计数器控件都会递增。如果我调整视图大小,列表框会显示我添加的所有项目。因此,ListBox控件上的ItemSource绑定被破坏。我也注意不要在任何时候创建一个新的MyCollection,这会破坏绑定。 Boo hoo。

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top