从两个单独的列表生成树视图,其中子对象具有父对象的 ID
Generate treeview from two seperate lists, where child objects has id to parent
我遇到了一个问题,我在 ViewModel 中有两个单独的 ObservableCollection。一个列表包含以下对象:
public class TypeCategory {
public int Id { get; set; }
public string Name { get; set; }
public string Description { get; set; }
}
另一个列表是:
public class Type {
public string Name { get; set; }
public string Description { get; set; }
public int CategoryId { get; set; }
}
ViewModel 就像:
public class TheViewModel {
public ObservableCollection<TypeCategory > TypeCategories { get; private set; }
public ObservableCollection<Type> Types { get; private set; }
}
Type
是 TypeCategory
的子节点,其中与父节点的连接是 CategoryId
。
是否可以通过使用例如多绑定和转换器来连接两个列表来生成 TreeView ?
或者是扩展父级 class 以包含其子级的唯一解决方案?
你可以这样试:
使用 Dictionary<string,List<string>>
,其中包含 TypeCategory Name
作为 Key
& List<string>
,其中包含在此 Category
下的 Type Names
。
public class TheViewModel
{
public ObservableCollection<TypeCategory> TypeCategories { get; set; }
public ObservableCollection<Type> Types { get; set; }
public Dictionary<string,List<string>> TreeViewItemSource { get; set; }
public TheViewModel()
{
TypeCategories = new ObservableCollection<TypeCategory>();
Types = new ObservableCollection<Type>();
TreeViewItemSource = new Dictionary<string, List<string>>();
for (int i = 0; i < 10; i++)
{
TypeCategories.Add(new TypeCategory() { Id = i, Name = "TypeCategory "+ i, Description = string.Empty });
}
for (int i = 0; i < 50; i++)
{
Types.Add(new Type() { CategoryId = i / 5, Name = "Type " + i, Description = string.Empty });
}
foreach (TypeCategory item in TypeCategories)
{
List<string> list = Types.Where(p => p.CategoryId == item.Id).Select(p=>p.Name).ToList<string>();
TreeViewItemSource.Add(item.Name, list);
}
}
}
XAML
:
<TreeView Name="treeView1" ItemsSource="{Binding TreeViewItemSource}" >
<TreeView.ItemTemplate>
<HierarchicalDataTemplate ItemsSource="{Binding Path=Value}">
<TextBlock FontWeight="Bold" Text="{Binding Path=Key}" />
<HierarchicalDataTemplate.ItemTemplate>
<DataTemplate>
<TextBlock Text="{Binding }"/>
</DataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
希望对您有所帮助。
UPDATE
:
如果您不想创建字典,则需要在 HierarchicalDataTemplate's ItemSource
上应用 MultiBinding
。
xmlns:local="clr-namespace:assembly_Name"
.....
<TreeView Name="treeView1" ItemsSource="{Binding TypeCategories}" Tag="{Binding Types}">
<TreeView.Resources>
<local:CategoryItemsConverter x:Key="CategoryItemsConverter"/>
</TreeView.Resources>
<TreeView.ItemTemplate>
<HierarchicalDataTemplate >
<HierarchicalDataTemplate.ItemsSource>
<MultiBinding Converter="{StaticResource CategoryItemsConverter}">
<Binding Path="Tag" ElementName="treeView1"/>
<Binding Path="ItemsSource" ElementName="treeView1"/>
<Binding Path="Name"/>
</MultiBinding>
</HierarchicalDataTemplate.ItemsSource>
<TextBlock FontWeight="Bold" Text="{Binding Name}" />
<HierarchicalDataTemplate.ItemTemplate>
<DataTemplate>
<TextBlock Text="{Binding Name}"/>
</DataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
CategoryItemsConverter :
public class CategoryItemsConverter : IMultiValueConverter
{
public object Convert(object[] values, System.Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
ObservableCollection<Type> typesList = values[0] as ObservableCollection<Type>;
ObservableCollection<TypeCategory> categoryList = values[1] as ObservableCollection<TypeCategory>;
string currentCategory = values[2] as string;
if (typesList != null && categoryList != null && !string.IsNullOrEmpty(currentCategory))
{
int? categoryId = categoryList.Where(p => p.Name == currentCategory).Select(p=>p.Id).FirstOrDefault();
if (categoryId.HasValue)
{
var a = typesList.Where(p => p.CategoryId == categoryId).ToList<Type>();
if (a.Any())
{
return a;
}
}
}
return null;
}
public object[] ConvertBack(object value, System.Type[] targetTypes, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException();
}
}
我遇到了一个问题,我在 ViewModel 中有两个单独的 ObservableCollection。一个列表包含以下对象:
public class TypeCategory {
public int Id { get; set; }
public string Name { get; set; }
public string Description { get; set; }
}
另一个列表是:
public class Type {
public string Name { get; set; }
public string Description { get; set; }
public int CategoryId { get; set; }
}
ViewModel 就像:
public class TheViewModel {
public ObservableCollection<TypeCategory > TypeCategories { get; private set; }
public ObservableCollection<Type> Types { get; private set; }
}
Type
是 TypeCategory
的子节点,其中与父节点的连接是 CategoryId
。
是否可以通过使用例如多绑定和转换器来连接两个列表来生成 TreeView ?
或者是扩展父级 class 以包含其子级的唯一解决方案?
你可以这样试:
使用 Dictionary<string,List<string>>
,其中包含 TypeCategory Name
作为 Key
& List<string>
,其中包含在此 Category
下的 Type Names
。
public class TheViewModel
{
public ObservableCollection<TypeCategory> TypeCategories { get; set; }
public ObservableCollection<Type> Types { get; set; }
public Dictionary<string,List<string>> TreeViewItemSource { get; set; }
public TheViewModel()
{
TypeCategories = new ObservableCollection<TypeCategory>();
Types = new ObservableCollection<Type>();
TreeViewItemSource = new Dictionary<string, List<string>>();
for (int i = 0; i < 10; i++)
{
TypeCategories.Add(new TypeCategory() { Id = i, Name = "TypeCategory "+ i, Description = string.Empty });
}
for (int i = 0; i < 50; i++)
{
Types.Add(new Type() { CategoryId = i / 5, Name = "Type " + i, Description = string.Empty });
}
foreach (TypeCategory item in TypeCategories)
{
List<string> list = Types.Where(p => p.CategoryId == item.Id).Select(p=>p.Name).ToList<string>();
TreeViewItemSource.Add(item.Name, list);
}
}
}
XAML
:
<TreeView Name="treeView1" ItemsSource="{Binding TreeViewItemSource}" >
<TreeView.ItemTemplate>
<HierarchicalDataTemplate ItemsSource="{Binding Path=Value}">
<TextBlock FontWeight="Bold" Text="{Binding Path=Key}" />
<HierarchicalDataTemplate.ItemTemplate>
<DataTemplate>
<TextBlock Text="{Binding }"/>
</DataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
希望对您有所帮助。
UPDATE
:
如果您不想创建字典,则需要在 HierarchicalDataTemplate's ItemSource
上应用 MultiBinding
。
xmlns:local="clr-namespace:assembly_Name"
.....
<TreeView Name="treeView1" ItemsSource="{Binding TypeCategories}" Tag="{Binding Types}">
<TreeView.Resources>
<local:CategoryItemsConverter x:Key="CategoryItemsConverter"/>
</TreeView.Resources>
<TreeView.ItemTemplate>
<HierarchicalDataTemplate >
<HierarchicalDataTemplate.ItemsSource>
<MultiBinding Converter="{StaticResource CategoryItemsConverter}">
<Binding Path="Tag" ElementName="treeView1"/>
<Binding Path="ItemsSource" ElementName="treeView1"/>
<Binding Path="Name"/>
</MultiBinding>
</HierarchicalDataTemplate.ItemsSource>
<TextBlock FontWeight="Bold" Text="{Binding Name}" />
<HierarchicalDataTemplate.ItemTemplate>
<DataTemplate>
<TextBlock Text="{Binding Name}"/>
</DataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
CategoryItemsConverter :
public class CategoryItemsConverter : IMultiValueConverter
{
public object Convert(object[] values, System.Type targetType, object parameter, System.Globalization.CultureInfo culture)
{
ObservableCollection<Type> typesList = values[0] as ObservableCollection<Type>;
ObservableCollection<TypeCategory> categoryList = values[1] as ObservableCollection<TypeCategory>;
string currentCategory = values[2] as string;
if (typesList != null && categoryList != null && !string.IsNullOrEmpty(currentCategory))
{
int? categoryId = categoryList.Where(p => p.Name == currentCategory).Select(p=>p.Id).FirstOrDefault();
if (categoryId.HasValue)
{
var a = typesList.Where(p => p.CategoryId == categoryId).ToList<Type>();
if (a.Any())
{
return a;
}
}
}
return null;
}
public object[] ConvertBack(object value, System.Type[] targetTypes, object parameter, System.Globalization.CultureInfo culture)
{
throw new NotImplementedException();
}
}