WPF DataGrid SelectedItem

本文关键字:SelectedItem DataGrid WPF | 更新日期: 2023-09-27 18:22:08

我有一个DataGrid,用户可以通过在最后一行输入数据来添加项目。我还有一个按钮,可以删除当前选择的项目。但是,当最后一行(为空,用于添加新项目)被选中时,最后一个被选中的项目将保留在SelectedItem中。因此,如果我打开窗口,选择最后一行,然后按下删除按钮,它将删除第一行,因为它是默认选择的,选择最后行不会更改SelectedItem。有什么好办法解决这个问题吗?

澄清:SelectedItem="{Binding X}"

当最后一行被选中时,ViewModel中的X不会改变(setter根本不会被调用)。我不确定SelectedItem属性本身是否发生了变化,但我认为它不会发生变化。

当我选择最后一行(红色边框)时也有一个例外,但当我再次单击它开始输入数据时,红色边框将消失。不确定这两者是否有关联。

WPF DataGrid SelectedItem

运行下面的示例,您会发现为什么它不起作用。

XAML:

<Window x:Class="DataGridTest.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525">
    <DockPanel>
        <TextBlock DockPanel.Dock="Bottom" Text="{Binding SelectedItem, ElementName=dataGrid}"/>
        <TextBlock DockPanel.Dock="Bottom" Text="{Binding SelectedItem}"/>
        <DataGrid x:Name="dataGrid" ItemsSource="{Binding Items}" SelectedItem="{Binding SelectedItem}" CanUserAddRows="True" CanUserDeleteRows="True" AutoGenerateColumns="False">
            <DataGrid.Columns>
                <DataGridTextColumn Header="First Name" Binding="{Binding FirstName}"/>
                <DataGridTextColumn Header="Last Name" Binding="{Binding FirstName}"/>
            </DataGrid.Columns>
        </DataGrid>
    </DockPanel>
</Window>

代码背后:

namespace DataGridTest
{
    using System.Collections.Generic;
    using System.Collections.ObjectModel;
    using System.ComponentModel;
    using System.Windows;
    public partial class MainWindow : Window, INotifyPropertyChanged
    {
        private readonly ICollection<Person> items;
        private Person selectedItem;
        public MainWindow()
        {
            InitializeComponent();
            this.items = new ObservableCollection<Person>();
            this.items.Add(new Person
                {
                    FirstName = "Kent",
                    LastName = "Boogaart"
                });
            this.items.Add(new Person
            {
                FirstName = "Tempany",
                LastName = "Boogaart"
            });
            this.DataContext = this;
        }
        public ICollection<Person> Items
        {
            get { return this.items; }
        }
        public Person SelectedItem
        {
            get { return this.selectedItem; }
            set
            {
                this.selectedItem = value;
                this.OnPropertyChanged("SelectedItem");
            }
        }
        private void OnPropertyChanged(string propertyName)
        {
            var handler = this.PropertyChanged;
            if (handler != null)
            {
                handler(this, new PropertyChangedEventArgs(propertyName));
            }
        }
        public event PropertyChangedEventHandler PropertyChanged;
    }
    public class Person
    {
        public string FirstName
        {
            get;
            set;
        }
        public string LastName
        {
            get;
            set;
        }
        public override string ToString()
        {
            return FirstName + " " + LastName;
        }
    }
}

正如你在跑步时所看到的;新的";行导致哨兵值被设置为DataGrid中的所选项目。但是,WPF无法将该sentinel项转换为Person,因此SelectedItem绑定无法转换。

要解决此问题,可以在绑定上放置一个转换器,用于检测sentinel并在检测到时返回null。这里有一个这样做的转换器:

namespace DataGridTest
{
    using System;
    using System.Windows.Data;
    public sealed class SentinelConverter : IValueConverter
    {
        public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        {
            return value;
        }
        public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        {
            if (item.Equals(CollectionView.NewItemPlaceholder)))
            {
                return null;
            }
            return value;
        }
    }
}

正如您所看到的,不幸的是,必须针对sentinel的ToString()值进行测试,因为它是一个内部类型。您也可以(或另外)检查GetType().Name是否为NamedObject

没有代码很难说,但我会考虑以下内容。

确保无论何时删除某个项并且该项也是选定项,都将绑定到ViewModel中属性的选定项设置为null。您需要确保绑定到属性的SelectedItem不是单向绑定的。

听起来好像忘记设置绑定模式,默认设置为OneWay。这意味着在视图中所做的任何更改都不会传播回视图模型。

并始终确保您拥有正确的数据上下文。

希望能有所帮助。