我将WPF Datagrid的ItemsSource属性绑定到ObservableCollection。数据被拉得很好,但是当我点击一个单元格(任何单元格)并开始使用键盘导航时,我注意到了奇怪的行为。
我相信我已将事件隔离到绑定的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");
}
}
}
}
答案 0 :(得分:0)
我知道这是一个较旧的问题,但最近我遇到了与DataGrid非常类似的问题。箭头键问题正如您所描述的那样。但是,我使用了DataGridTemplateColumns,其中TextBox允许编辑(否则,正如您所说,我无法编辑数据)。如果我编辑了一个单元格然后单击它(特别是如果我单击它上面的行),DataGrid通常会产生一个重复的行(尽管集合中的计数保持不变)。
就我而言,我在我的Equals(object obj)方法中发现了一个错误,该类的成员在ObservableCollection中被绑定到DataGrid。这阻止了对Equals的调用确定两个项目之间的真正相等性,并且这在某种程度上导致了大问题。例如,如果在编辑一行后,数据网格无法在其列表中找到相同的行(通过调用错误的Equals方法),也许它会生成另一行(假设它不能为该条目设置一行)
无论如何,我建议你在LapViewModel中查看Equals方法。
希望有所帮助。