WPF datagrid箭头(右/左)键导航异常

时间:2011-01-17 15:26:46

标签: c# wpf datagrid

我将WPF Datagrid的ItemsSource属性绑定到ObservableCollection。数据被拉得很好,但是当我点击一个单元格(任何单元格)并开始使用键盘导航时,我注意到了奇怪的行为。

  1. Tab键按预期工作(从左到右,并包裹到下一行)。
  2. 向上键不执行任何操作(焦点停留在所选单元格上)。
  3. 向下键将焦点移动到列中的顶部单元格。例如,如果我在B列的第10行并点击“向下”,则焦点单元格将成为B列的第0行。
  4. 左键或右键会导致ArgumentOutOfRangeException。 “指定的参数超出了有效值的范围。参数名称:索引”。没有InnerException,Stack Trace也没有多大帮助。
  5. 双击任何单元格都不会启用编辑模式。
  6. 由于某种原因,我在网格底部获得了一个“额外”行(rowcount =(集合计数+1)),该行具有正确的功能(右/左按钮工作,双击触发编辑模式)。当我双击进入此额外行中的编辑模式,然后单击其上方的绑定行时,会在网格中添加一行。
  7. 我相信我已将事件隔离到绑定的ViewModel类(而不是XAML)。在初步调试中,我已经将XAML剥离到最低限度(没有样式,2列,只有绑定是ItemsSource)并且仍然得到奇怪的行为。

    <UserControl x:Name="FuelMileageView" x:Class="TRD.RaceStrategy.Views.FuelMileage"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:TRD.RaceStrategy"
             xmlns:views="clr-namespace:TRD.RaceStrategy.Views"
             xmlns:vm="clr-namespace:TRD.RaceStrategy.ViewModels;assembly=RaceStrategy.Support"
             xmlns:converters="clr-namespace:TRD.RaceStrategy.Converters;assembly=RaceStrategy.Support"
             xmlns:Behaviours="clr-namespace:TRD.RaceStrategy.Behaviours;assembly=RaceStrategy.Support"
             mc:Ignorable="d" DataContext="{Binding Path=Properties[PrimaryVM].CarEventVM.FuelMileage.FuelMileageLaps, Source={x:Static local:App.Current}}"
             d:DesignHeight="410" d:DesignWidth="485">
    
    <DataGrid x:Name="dgFMLaps" ItemsSource="{Binding}"
                AutoGenerateColumns="False">
        <DataGrid.Columns>
            <DataGridTextColumn Header="Column 1" >
            </DataGridTextColumn>
            <DataGridTextColumn Header="Column 2" >
            </DataGridTextColumn>
        </DataGrid.Columns>            
    </DataGrid>
    </UserControl>
    

    除了InitializeComponent()调用之外,没有代码隐藏可以说它似乎将绑定的FuelMileageLaps集合作为唯一的罪魁祸首。

    public class FuelMileageLapViewModel : LapViewModel
    {
        public FuelMileageLapViewModel() { }    
    }
    

    注意:ObservableCollectionEx是ObservableCollection类的扩展,它显然解决了线程问题(?)我已将此类与其他集合一起使用,而这些集合又插入到没有此键盘导航问题的数据网格中。

    public class ObservableCollectionEx<T> : ObservableCollection<T>
    {
        // Override the event so this class can access it
        public override event System.Collections.Specialized.NotifyCollectionChangedEventHandler CollectionChanged;
    
        protected override void OnCollectionChanged(System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
        {
            // Be nice - use BlockReentrancy like MSDN said
            using (BlockReentrancy())
            {
                System.Collections.Specialized.NotifyCollectionChangedEventHandler eventHandler = CollectionChanged;
                if (eventHandler == null)
                    return;
    
                Delegate[] delegates = eventHandler.GetInvocationList();
                // Walk thru invocation list
                foreach (System.Collections.Specialized.NotifyCollectionChangedEventHandler handler in delegates)
                {
                    DispatcherObject dispatcherObject = handler.Target as DispatcherObject;
                    // If the subscriber is a DispatcherObject and different thread
                    if (dispatcherObject != null && dispatcherObject.CheckAccess() == false)
                    {
                        // Invoke handler in the target dispatcher's thread
                        dispatcherObject.Dispatcher.Invoke(DispatcherPriority.DataBind, handler, this, e);
                    }
                    else // Execute handler as is
                        handler(this, e);
                }
            }
        }
    }
    

    在这一点上,我的下一步是关于调试的下一步。在断点或try / catch块中没有明显的位置。我用Google搜索了我的问题,并没有找到任何有价值的东西。请帮忙!

    这是FuelMileageViewModel类,其中初始化了FuelMileageLaps:

    public class FuelMileageViewModel : WorkspaceViewModel
    {
        /// <summary>
        /// View model for the fuel mileage calculator
        /// </summary>
        /// <param name="car">Car on which to base all calculations</param>
        public FuelMileageViewModel()
        {}
    
        /// <summary>
        /// A separate collection of laps that store the extra fuel mileage data
        /// </summary>
        public ObservableCollectionEx<FuelMileageLapViewModel> FuelMileageLaps
        {
            get
            {
                if (_fuelMileageLaps == null)
                {
                    _fuelMileageLaps = new ObservableCollectionEx<FuelMileageLapViewModel>();
                }
                return _fuelMileageLaps;
            }
            set
            {
                _fuelMileageLaps = value;
                OnPropertyChanged("FuelMileageLaps");
            }
        }
        private ObservableCollectionEx<FuelMileageLapViewModel> _fuelMileageLaps;
    
        /// <summary>
        /// Number of laps in the race
        /// Affects: Laps_G, Laps_Y
        /// </summary>
        public int NumberOfLaps
        {
            get
            {
                return FuelMileageLaps.Count;
            }
            set
            {
                int count = FuelMileageLaps.Count;
    
                if (value < 0)
                {
                    throw new ArgumentException("Number of laps must be a positive integer");
                }
    
                if (count != value)
                {
                    if( count < value )
                    {
                        int diff = value - count;
                        for (int i = 0; i < diff; i++)
                        {
                            FuelMileageLapViewModel lapToAdd = new FuelMileageLapViewModel();
    
                            FuelMileageLaps.Add(lapToAdd);
                        }
                    }
                    OnPropertyChanged("NumberOfLaps");
                }
            }
        }
    }
    

1 个答案:

答案 0 :(得分:0)

我知道这是一个较旧的问题,但最近我遇到了与DataGrid非常类似的问题。箭头键问题正如您所描述的那样。但是,我使用了DataGridTemplateColumns,其中TextBox允许编辑(否则,正如您所说,我无法编辑数据)。如果我编辑了一个单元格然后单击它(特别是如果我单击它上面的行),DataGrid通常会产生一个重复的行(尽管集合中的计数保持不变)。

就我而言,我在我的Equals(object obj)方法中发现了一个错误,该类的成员在ObservableCollection中被绑定到DataGrid。这阻止了对Equals的调用确定两个项目之间的真正相等性,并且这在某种程度上导致了大问题。例如,如果在编辑一行后,数据网格无法在其列表中找到相同的行(通过调用错误的Equals方法),也许它会生成另一行(假设它不能为该条目设置一行)

无论如何,我建议你在LapViewModel中查看Equals方法。

希望有所帮助。