如何在过滤 ICollectionView 后选择第一个项目

4
我正在将一个ListView绑定到我的视图模型中的ICollectionView。当您单击某些按钮时,ICollectionView具有一些预定义的过滤器。但是,我似乎找不到任何方法在集合被过滤后(自动)选择ListView中的第一项。
我尝试设置SelectedIndex=0,将Target和Source通知都添加到绑定中,但当应用筛选器时,所有这些都无效。
有什么指针可以帮助我实现这个目标吗?
编辑:下面的代码说明了我的问题。
XAML:
<Window x:Class="CollectionViewTest.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:CollectionViewTest"
        mc:Ignorable="d"
        Title="MainWindow" Height="450" Width="800">

    <Window.DataContext>
        <local:MainViewModel/>
    </Window.DataContext>
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition />
            <ColumnDefinition />
            <ColumnDefinition />
        </Grid.ColumnDefinitions>

        <!-- MENU -->
        <StackPanel Orientation="Vertical">
            <Button Content="Numbers below 4" Click="Below4_Click" Width="100"/>
            <Button Content="Numbers below 7" Click="Below7_Click" Width="100"/>
            <Button Content="All numbers" Click="All_Click" Width="100"/>
        </StackPanel>

        <!-- LIST -->
        <ListView 
            Grid.Column="1" 
            SelectedIndex="0"
            ItemsSource="{Binding Numbers, Mode=OneWay}"
            SelectedItem="{Binding SelectedNumber, Mode=TwoWay}">
            <ListView.Resources>
                <DataTemplate DataType="{x:Type local:Number}">
                    <TextBlock Text="{Binding Value}" />
                </DataTemplate>
            </ListView.Resources>
        </ListView>

        <!-- DETAILS -->
        <TextBlock Grid.Column="2" Text="{Binding SelectedNumber.Text}" Width="100"/>
    </Grid>
</Window>

代码后置:

using System.Windows;

namespace CollectionViewTest
{
    public partial class MainWindow : Window
    {
        private MainViewModel vm;
        public MainWindow()
        {
            InitializeComponent();
            vm = (MainViewModel)DataContext;
        }

        private void Below4_Click(object sender, RoutedEventArgs e)
        {
            vm.MenuFilter = f => f.Value < 4;
        }

        private void Below7_Click(object sender, RoutedEventArgs e)
        {
            vm.MenuFilter = f => f.Value < 7;
        }

        private void All_Click(object sender, RoutedEventArgs e)
        {
            vm.MenuFilter = f => true;
        }
    }
}

视图模型:

using System;
using System.ComponentModel;
using System.Runtime.CompilerServices;
using System.Windows.Data;
using System.Collections.ObjectModel;

namespace CollectionViewTest
{
    public class MainViewModel : PropertyChangedBase
    {

        public MainViewModel()
        {
            Numbers = new ObservableCollection<Number>();
            NumberCollection = CollectionViewSource.GetDefaultView(Numbers);
            NumberCollection.Filter = Filter;
            NumberCollection.SortDescriptions.Add(new SortDescription("Value", ListSortDirection.Ascending));

            for (int i = 0; i < 10; i++)
                Numbers.Add(new Number { Value = i, Text = $"This is number {i}." });

        }

        private Func<Number, bool> menuFilter;
        public Func<Number, bool> MenuFilter
        {
            get => menuFilter;
            set
            {
                menuFilter = value;
                NumberCollection.Refresh();
            }
        }

        private bool Filter(object item)
        {
            var number = (Number)item;
            return MenuFilter == null ? true : MenuFilter(number);
        }

        public ObservableCollection<Number> Numbers { get; set; }
        public ICollectionView NumberCollection { get; set; }

        private Number selectedNumber;
        public Number SelectedNumber { get => selectedNumber; set => Set(ref selectedNumber, value); }
    }

    public class Number : PropertyChangedBase
    {
        public int Value { get; set; }

        private string text;
        public string Text { get => text; set => Set(ref text, value); }
    }

    public class PropertyChangedBase : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;

        protected void Set<T>(ref T field, T newValue = default(T), [CallerMemberName] string propertyName = null)
        {
            field = newValue;
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
        }
    }
}

从上面的按钮可以看出,按下其中一个按钮会改变过滤器并调用集合的刷新。我希望的是,自动选择列表中的第一项(这里是“0”),然后在第二列的文本中显示“这是数字0”。

我已经尝试了SelectedIndex = 0和MoveCurrentToFirst,但没有选中任何内容。


1
你真的尝试将ListView的SelectedIndex属性设置为0了吗?这应该可以工作。 - mm8
是的,当筛选条件改变时它就不起作用了。我会尝试在周末在我的上下文之外进行测试,但我在这里的应用程序中使用的是相当基本的 ICollectionView + ListView,所以我有些怀疑...但我会回来的。 - Werner
@mm8 示例已添加。如果我不调用ICollectionView.Refresh,则似乎设置SelectedIndex = 0只起作用。但是我必须这样做才能评估过滤器? - Werner
1个回答

9
不要在绑定到 ICollectionView 时设置 SelectedIndex。相反,通过 MoveCurrentTo()MoveCurrentToFirst() 设置其 CurrentItem
myCollectionView.MoveCurrentTo(someItem);
...
myCollectionView.MoveCurrentToFirst();

此外,在您的ListView上设置IsSynchronizedWithCurrentItem属性:
<ListView IsSynchronizedWithCurrentItem="True" ...

检测筛选器应用

当筛选器被评估时,集合视图会刷新,从而重置集合。要检测这一点,请监听CollectionChanged事件,并查找NotifyCollectionChangedAction.Reset标志。有关更多详细信息,请参阅CollectionView源代码


1
myCollectionView.MoveCurrentToFirst() - Rekshino
但是MoveCurrentToFirst和派生方法是我需要调用以进行更改的内容 - 而不是设置。据我所知,我无法知道何时完成过滤? - Werner
@l33t 请看一下我的更新示例,我尝试订阅CollectionChanged事件(不在示例中)。但是调用MoveCurrentToFirst()似乎没有任何作用。 - Werner
@l33t 哎呀 - 我忘记了 IsSynchronizedWithCurrentItem,然后它似乎可以工作!让我再做一些测试... - Werner
1
@l33t,没错,就是这个。LisView需要IsSynchronizedWithCurrentItem。订阅CollectionChanged事件并调用MoveCurrentToFirst方法。有点复杂,但非常有效。谢谢。 - Werner

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接