WPF - Grouped CollectionView多次显示同一组

时间:2017-02-27 15:42:33

标签: wpf xaml data-binding grouping collectionview


我在分配绑定到ItemsControl的CollectionView时遇到问题。

在我的ViewModel中,我有多个组。 Group是Person的集合,每个Person都知道它属于哪个组 最后查看组和人员类。

  - 这就是我想要做的事情:
我希望能够显示按组分组的人员集合。我强调我想在视图中从人员集合创建这些组,不是来自群组。·
  - 这是我怎么做的:
我有一个列表,其中包含所有组中的每个人(列表名为" Everyone"),我有一个我从Everyone创建的CollectionView,名为" EveryoneGrouped"。然后我添加一个从" Group"创建的PropertyGroupDescription;到我的EveryoneGrouped集合。

  - 这是我的问题:
组出现多次,实际上组出现的次数与其包含的人数一样多 如果我有以下组:
  - 第一组[" Alpha One"]
  - 第二组[" Alpha Two"," Beta Two"]
  - 第三组[" Alpha III"," Beta 3"," Gamma Three"]
它将产生以下结果: Application Screenshot
第一组只出现一次,因为它只包含一个人。我希望第二组和第三组也只出现一次 我无法指出我做错了什么,我们将非常感谢任何帮助。

修改:多个群组或人员可以使用相同的名称。

这是我的代码,它生成了上一个屏幕截图:

视图模型

public class ViewModel
{
    public CollectionView EveryoneGrouped { get; private set; }
    private List<Person> Everyone { get; set; } = new List<Person>();
    private List<Group> AllGroups { get; set; } = new List<Group>();

    public ViewModel()
    {
        populateGroups();
        populateEveryoneCollection();
        createCollectionView();
    }

    private void populateGroups()
    {
        Group one = new Group
        {
            new Person { PersonName = "Alpha One" }
        };
        one.GroupName = "ONE";

        Group two = new Group
        {
            new Person { PersonName = "Alpha Two" },
            new Person { PersonName = "Beta Two" }
        };
        two.GroupName = "TWO";

        Group three = new Group
        {
            new Person { PersonName = "Alpha Three" },
            new Person { PersonName = "Beta Three" },
            new Person { PersonName = "Gamma Three" }
        };
        three.GroupName = "THREE";

        AllGroups.Add(one);
        AllGroups.Add(two);
        AllGroups.Add(three);
    }

    private void populateEveryoneCollection()
    {
        foreach(Group group in AllGroups)
        {
            foreach(Person person in group)
            {
                Everyone.Add(person);
            }
        }
    }

    private void createCollectionView()
    {
        EveryoneGrouped = (CollectionView)CollectionViewSource.GetDefaultView(Everyone);
        PropertyGroupDescription groupDescription = new PropertyGroupDescription("Group");
        EveryoneGrouped.GroupDescriptions.Add(groupDescription);
    }
}

XAML

<Window x:Class="FunWithCollectionViewGrouping.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:FunWithCollectionViewGrouping"
    mc:Ignorable="d"
    Title="MainWindow" Height="350" Width="525">
<Grid>
    <ItemsControl ItemsSource="{Binding EveryoneGrouped}">

        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>
                <StackPanel Orientation="Horizontal"/>
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>

        <ItemsControl.ItemTemplate>
            <DataTemplate>
                <TextBlock Text="{Binding PersonName}" Margin="5,0"/>
            </DataTemplate>
        </ItemsControl.ItemTemplate>

        <ItemsControl.GroupStyle>
            <GroupStyle>
                <GroupStyle.HeaderTemplate>
                    <DataTemplate>
                        <TextBlock DockPanel.Dock="Left" FontWeight="Bold" Text="{Binding Name.Group.GroupName}" />
                    </DataTemplate>
                </GroupStyle.HeaderTemplate>
            </GroupStyle>
        </ItemsControl.GroupStyle>
    </ItemsControl>
</Grid>

代码背后

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        DataContext = new ViewModel();
    }
}

public class Person
{
    private static int PersonQuantity = 0;
    private int _id = ++PersonQuantity;
    public int Id { get { return _id; } }
    public Group Group { get; set; }
    public string PersonName { get; set; }
}

public class Group : ObservableCollection<Person>
{
    private static int GroupQuantity = 0;
    private int _id = ++GroupQuantity;
    public int Id { get { return _id; } }
    public string GroupName { get; set; }

    public Group()
    {
        CollectionChanged += Group_CollectionChanged;
    }

    private void Group_CollectionChanged(object sender, System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
    {
        if( e.NewItems != null )
        {
            foreach(Person p in e.NewItems)
            {
                if ( p.Group != null ) { p.Group.Remove(p); }
                p.Group = this;
            }
        }

        if( e.OldItems != null )
        {
            foreach (Person p in e.OldItems)
            {
                if (p.Group != null && p.Group.Equals(this)) { p.Group = null; }
            }
        }
    }

    public override bool Equals(object obj)
    {
        Group other = obj as Group;
        if (obj == null) return false;
        return Id == other.Id;
    }

    public override int GetHashCode()
    {
        return Id;
    }
}

谢谢。

1 个答案:

答案 0 :(得分:3)

GroupName的{​​{1}}属性分组:

Group

并显示该组的名称:

private void createCollectionView()
{
    EveryoneGrouped = (CollectionView)CollectionViewSource.GetDefaultView(Everyone);
    PropertyGroupDescription groupDescription = new PropertyGroupDescription("Group.GroupName");
    EveryoneGrouped.GroupDescriptions.Add(groupDescription);
}
  

我对此解决方案的问题是多个组可以具有相同的名称(我更新了我的帖子以指定它),如果我这样做,它们将在视图中合并。组的ID由它们的ID区分(我不能排序,因为我会在视图中松开组名称。)

<ItemsControl.GroupStyle> <GroupStyle> <GroupStyle.HeaderTemplate> <DataTemplate> <TextBlock DockPanel.Dock="Left" FontWeight="Bold" Text="{Binding Name}" /> </DataTemplate> </GroupStyle.HeaderTemplate> </GroupStyle> </ItemsControl.GroupStyle> 属性分组:

Id

并将private void createCollectionView() { EveryoneGrouped = (CollectionView)CollectionViewSource.GetDefaultView(Everyone); PropertyGroupDescription groupDescription = new PropertyGroupDescription("Group.Id"); EveryoneGrouped.GroupDescriptions.Add(groupDescription); } 绑定到组中第一项TextBlock的{​​{1}}属性:

GroupName
  

我不明白为什么它表现得这样,PropertyGroupDescription不使用Equals和GetHashCode?

显然不是。它使用反射能够按字符串指定的任何属性进行分组。