我有两个 ObservableCollections填充了相同类型的Items(Items也是ObservableCollecions,其中包含其他类型的对象)。
_TreeViewBase - 第一个ObservableCollection是' base'所有数据。
_TVData - 第二个ObservableCollection只应存储'需要'项目,我还有TreeView与ItemSource绑定到TVData。
函数 FillAliasTreeView 仅用于使用数据填充_TreeViewBase,因此我不会在此处粘贴代码。
在程序开始时,将
但是当我搜索某些东西(并且我的TVData在 SearchBox_TextChanged()中被修改)时,它似乎是 还修改了_TreeViewBase。
怎么可能?我在代码中犯了一些重大错误吗?
我的课程(几乎完整的代码):
public partial class SideBar : Page, INotifyPropertyChanged
{
private ObservableCollection<ConnectionModel> _TreeViewBase;
private ObservableCollection<ConnectionModel> _TVData ;
public SideBar()
{
TreeViewBase = new ObservableCollection<ConnectionModel>( ConnectionUtilities.LoadObservableConnections() ) ;
TVData = new ObservableCollection<ConnectionModel>();
InitializeComponent();
DataContext = this;
var loadSideBar = Task.Factory.StartNew( async () => { await LoadSideBar(); } );
loadSideBar.Wait();
OnPropertyChanged("TVData");
}
public async Task LoadSideBar()
{
// this function is used to fill my _TreeViewBase
await FillAliasTreeView();
TVData = TreeViewBase;
OnPropertyChanged("TVData");
}
private async void SearchBox_TextChanged(object sender, TextChangedEventArgs e)
{
string newsearchText = SearchBox.Text;
if ( (newsearchText == "" || newsearchText == "Search..." || newsearchText.Length < 3 ) )
{
TVData.Clear();
TVData = null;
MessageBox.Show( (TVData == null).ToString() + " " + TreeViewBase.Count.ToString());
TVData = new ObservableCollection<ConnectionModel>( TreeViewBase );
AliasTree.ItemsSource = TVData ;
AliasTree.UpdateLayout();
return;
}
searchText = SearchBox.Text;
if (TVData != null)
{
await Dispatcher.InvokeAsync(
() =>
{
foreach (ConnectionModel cm in TVData.ToList<ConnectionModel>())
{
foreach (SchemaModel sm in cm.schemas.ToList<SchemaModel>())
{
foreach (SchemaCollection sc in sm.schema_collections.ToList<SchemaCollection>())
{
try
{
ObservableCollection<TableModel> octm = (ObservableCollection<TableModel>)sc.collection;
for (int i = 0; i < octm.Count; i++)
{
if (!(sc.collection as ObservableCollection<TableModel>)[i].TABLE_NAME.Contains(searchText))
{
(sc.collection as ObservableCollection<TableModel>).RemoveAt(i);
i = i - 1;
}
Dispatcher.InvokeAsync( () => OnPropertyChanged("tables") ) ;
}
Dispatcher.InvokeAsync(() => OnPropertyChanged("collection") );
}
catch (Exception exc) { }
}
Dispatcher.InvokeAsync(() => OnPropertyChanged("schemas") );
}
Dispatcher.InvokeAsync(() => OnPropertyChanged("TVData"));
}
Dispatcher.InvokeAsync(() => OnPropertyChanged("TVData"));
});
await Dispatcher.InvokeAsync(() => ExpandAll(AliasTree));
}
}
private void ExpandAll( ItemsControl root )
{
if ( root != null )
{
foreach (var subItem in root.Items)
{
if (subItem != null)
{
ExpandAll( (TreeViewItem)root.ItemContainerGenerator.ContainerFromItem(subItem) );
try
{
if (((TreeViewItem)root.ItemContainerGenerator.ContainerFromItem(subItem) as TreeViewItem) != null )
((TreeViewItem)root.ItemContainerGenerator.ContainerFromItem(subItem) as TreeViewItem).IsExpanded = true;
}
catch (Exception exc) { };
}
}
AliasTree.UpdateLayout();
}
}
感谢您提出任何建议。
答案 0 :(得分:5)
在构造函数中,您将TVData
和TreeViewBase
设置为ObservableCollection<>
的两个不同实例。但接下来这一行:
TVData = TreeViewBase;
使它们引用相同的实例。旧的TVData
可观察集合将被丢弃,这两个变量现在都引用相同的可观察集合。