当 CompositeCollection 的当前位置发生变化时,有没有办法得到通知?
我需要由 CollectionView 监控 CompositeCollection,欢迎提出任何想法。
当 CompositeCollection 的当前位置发生变化时,有没有办法得到通知?
我需要由 CollectionView 监控 CompositeCollection,欢迎提出任何想法。
ICollectionView.CurrentChanged
您可以通过监视CollectionView的事件来检测当前项目何时更改。以下代码适用于我:
CompositeCollection cc = new CompositeCollection();
cc.Add(new CollectionContainer { Collection = new string[] { "Oh No!", "Fie" } });
cc.Add(new CollectionContainer { Collection = new string[] { "Zounds", "Ods Bodikins" } });
CollectionViewSource cvs = new CollectionViewSource { Source = cc };
// Subscribing to CurrentChanged on the ICollectionView
cvs.View.CurrentChanged += (o, e) => MessageBox.Show("current changed");
lb.ItemsSource = cvs.View; // lb is a ListBox with IsSynchronizedWithCurrentItem="True"
当我更改 ListBox 中的选择时,会显示消息框。
关于过滤、排序和分组,根据 Aron 的回答,这些在 CompositeCollection 上的视图中不可用。但为了记录,您可以通过以下方式检测支持这些功能的视图的更改:
ObservableCollection<GroupDescription>
,因此在 GroupDescriptions 属性上连接一个 CollectionChanged 事件处理程序。您不能在 copmposite 集合上运行 CollectionView,请参见此处
我遇到了同样的问题:我需要对 CompositeCollection 进行排序。我编写了以下解决问题的类,至少对于相同类型的 ObservableCollections。
这个想法是将复合集合维护为普通的可观察集合,并随着基础集合的变化对其进行更新。然后生成的集合(AllNodes)可以在用户界面中使用,它支持 CollectionView 就好了。
using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.Linq;
namespace Util {
public class ObservableCollectionCollector<T> {
private class ReplacableObservableCollection : ObservableCollection<T> {
public void Replace(int idx, T v) {
SetItem(idx, v);
}
}
private readonly ReplacableObservableCollection allNodes;
private readonly ObservableCollection<T>[] colls;
private readonly int[] lens;
public ObservableCollectionCollector(params ObservableCollection<T>[] colls) {
this.colls = colls;
allNodes = new ReplacableObservableCollection();
foreach (var l in colls) {
foreach (var e in l)
allNodes.Add(e);
l.CollectionChanged += HandleCollectionChanged;
}
lens = colls.Select(c => c.Count).ToArray();
}
public ReadOnlyObservableCollection<T> AllNodes {
get { return new ReadOnlyObservableCollection<T>(allNodes); }
}
private void HandleCollectionChanged(object sender, NotifyCollectionChangedEventArgs e) {
int i0 = 0;
int ci = 0;
foreach (var l in colls) {
if (l == sender)
break;
i0 += l.Count;
++ci;
}
switch (e.Action) {
case NotifyCollectionChangedAction.Add:
for (int i = 0; i < e.NewItems.Count; ++i)
allNodes.Insert(i0 + e.NewStartingIndex + i, (T)e.NewItems[i]);
break;
case NotifyCollectionChangedAction.Move:
for (int i = 0; i < e.OldItems.Count; ++i)
allNodes.Move(i0 + e.OldStartingIndex + i, i0 + e.NewStartingIndex + i);
break;
case NotifyCollectionChangedAction.Remove:
for (int i = 0; i < e.OldItems.Count; ++i)
allNodes.RemoveAt(i0 + e.OldStartingIndex);
break;
case NotifyCollectionChangedAction.Replace:
for (int i = 0; i < e.NewItems.Count; ++i)
allNodes.Replace(i0 + e.OldStartingIndex + i, (T)e.NewItems[i]);
break;
case NotifyCollectionChangedAction.Reset:
for (int i = 0; i < lens[ci]; ++i)
allNodes.RemoveAt(i0);
break;
}
lens[ci] = ((ObservableCollection<T>)sender).Count;
}
}
}