我想将值和列索引传递给一个方法,该方法将以编程方式选择DataGrid控件中与给定列中的值匹配的行。
我的代码是这样的:
private void HighlightSelections(string selection, int colIndex)
{
mtoDG.UnselectAll();
for(int i = 0; i < mtoDG.Items.Count; i++)
{
DataGridRow row = mtoDG.ItemContainerGenerator.ContainerFromIndex(i) as DataGridRow;
if (mtoDG.Columns[colIndex].GetCellContent(row) is TextBlock cellContent && cellContent.Text.Equals(selection))
{
object item = mtoDG.Items[i];
mtoDG.SelectedItems.Add(item);
}
}
}
我发现只有在屏幕上显示整个数据网格时,此方法才有效。如果由于空间约束而有任何未显示的行,那么它将抛出一个nullexception错误。
所以我的问题是,即使显示区域中有看不见的行,我的代码中是否有任何可以改变它以使其工作?
首先,通过添加ArgumentNullException
来处理row != null
:
DataGridRow row = mtoDG.ItemContainerGenerator.ContainerFromIndex(i) as DataGridRow;
if (row != null)
{
if (mtoDG.Columns[colIndex].GetCellContent(row) is TextBlock cellContent && cellContent.Text.Equals(selection))
{
object item = mtoDG.Items[i];
mtoDG.SelectedItems.Add(item);
}
}
其次,订阅ItemContainerGenerator.StatusChanged
事件以刷新HighlightSelections
:
mtoDG.ItemContainerGenerator.StatusChanged += ItemContainerGenerator_StatusChanged;
private void ItemContainerGenerator_StatusChanged(object sender, EventArgs e)
{
// HighlightSelections(?, ?);
}
这里的好解决方案是让DataContext具有行的IsSelected属性,然后你应该只将它与行IsSelected绑定,之后你可以设置你的DataContext属性,一切都应该没问题,因为你的DataContext总是有有效的项目。
我想你需要先找出导致ArgumentNullException的原因。禁用DataGrid虚拟化功能可能会有所帮助。