WPF数据网格循环/选择具有特定属性的单元格

时间:2015-10-13 19:34:56

标签: .net wpf binding datagrid

WPF全新,对WinForms非常熟悉(可能会使转换变得更加粗糙)。我正在尝试将旧的WinForms项目中的一些功能移植到WPF作为学习体验。

目标是在DataGrid中查找与TextBox中的字符串匹配的单元格值。我找到了一个great example使用绑定来完成它。基本上链接的代码会将任何匹配的DataGridCell的背景颜色更改为橙​​色。我已经修改了我的版本,但功能应该是相同的。请参阅代码示例的链接,这里提供它似乎有点多余。填充我的DataGrid的数据来自DataTable(如果重要的话)。

我想要做的是有一个“下一步”按钮,它将遍历每个单元格(通过使用背景颜色或自定义属性DataGridTextSearch.IsTextMatch确定)并选择它。似乎可以只修改一些提供的代码,但我不知道从哪里开始。在我的旧WinForms项目中,我将DataGridViewCell存储在一个列表中(在使用Linq查询找到它们之后)并且只是附加了按钮行为以递增所述列表并设置当前单元格。我怀疑可能有更聪明/更好的方式涉及绑定,我甚至不知道如何将这些匹配的单元格添加到列表中,如果这是一个选项。

因此,总而言之,我想要一个循环遍历特定DataGridCells的按钮(基于Background或自定义DataGridTextSearch.IsTextMatch属性)并选择它们。

提前致谢。

1 个答案:

答案 0 :(得分:5)

根据您在问题中提供的link,我已经找到了解决方法。在我的解决方案中,当DataGridCellTextBox中的字符串匹配时,它的Tag属性将设置为“1”,然后当Button被点击时,它将会迭代所有DataGridCells并查找具有非空Tags的项目,最后突出显示的单元格将逐个聚焦。

这是一个给你一个想法的工作示例:

<强>的Xaml:

<Window Name="UI">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="*"/>
            <RowDefinition Height="Auto"/>
        </Grid.RowDefinitions>
        <StackPanel DataContext="{Binding ElementName=UI}" Grid.Row="0">
            <TextBox Name="SearchBox" TextChanged="SearchBox_TextChanged"/>
            <DataGrid x:Name="grid" 
                  m:DataGridTextSearch.SearchValue="{Binding ElementName=SearchBox, Path=Text, UpdateSourceTrigger=PropertyChanged}" 
                  ItemsSource="{Binding TestData}"
                  SelectionUnit="Cell">
                <DataGrid.Resources>
                    <m:SearchValueConverter x:Key="SearchValueConverter" />
                    <Style TargetType="{x:Type DataGridCell}">
                        <Setter Property="m:DataGridTextSearch.IsTextMatch">
                            <Setter.Value>
                                <MultiBinding Converter="{StaticResource SearchValueConverter}">
                                    <Binding RelativeSource="{RelativeSource Self}" Path="Content.Text" />
                                    <Binding RelativeSource="{RelativeSource Self}" Path="(m:DataGridTextSearch.SearchValue)" />
                                </MultiBinding>
                            </Setter.Value>
                        </Setter>
                        <Style.Triggers>
                            <Trigger Property="m:DataGridTextSearch.IsTextMatch" Value="True">
                                <Setter Property="Background" Value="Orange" />
                                <Setter Property="Tag" Value="1" />
                            </Trigger>
                        </Style.Triggers>
                    </Style>
                </DataGrid.Resources>
            </DataGrid>
        </StackPanel>
        <Button Grid.Row="1" Click="Button_Click" Content="GoNext"/>
    </Grid>
</Window>

<强> MainWindow.cs:

int currentIndex = 0;

private void SearchBox_TextChanged(object sender, TextChangedEventArgs e)
{
    currentIndex = 0;
}

private void Button_Click(object sender, RoutedEventArgs e)
{
    var selectedCells = GetHighLightedCells();
    if (selectedCells.Count == 0)
        return;

    selectedCells[currentIndex].Focus();

    if (currentIndex == selectedCells.Count - 1)
        currentIndex = 0;
    else
        currentIndex++;
}

获取突出显示的单元格的方法:

public List<DataGridCell> GetHighLightedCells()
{
    List<DataGridCell> selectedCells = new List<DataGridCell>();
    foreach (DataGridRow rowContainer in GetDataGridRows())
    {
        if (rowContainer != null)
        {
            DataGridCellsPresenter presenter = GetVisualChild<DataGridCellsPresenter>(rowContainer);
            foreach (var col in grid.Columns)
            {
                DataGridCell cell = (DataGridCell)presenter.ItemContainerGenerator.ContainerFromIndex(col.DisplayIndex);
                if (cell == null)
                {
                    grid.ScrollIntoView(rowContainer, col);
                    cell = (DataGridCell)presenter.ItemContainerGenerator.ContainerFromIndex(col.DisplayIndex);
                }
                if (cell.Tag != null)
                {
                    selectedCells.Add(cell);
                }
            }
        }
    }
    return selectedCells;
}
public IEnumerable<DataGridRow> GetDataGridRows()
{
    var itemsSource = grid.ItemsSource as IEnumerable;
    if (null == itemsSource) yield return null;
    foreach (var item in itemsSource)
    {
        var row = grid.ItemContainerGenerator.ContainerFromItem(item) as DataGridRow;
        if (null != row) yield return row;
    }
}

public static T GetVisualChild<T>(Visual parent) where T : Visual
{
    T child = default(T);
    int numVisuals = VisualTreeHelper.GetChildrenCount(parent);
    for (int i = 0; i < numVisuals; i++)
    {
        Visual v = (Visual)VisualTreeHelper.GetChild(parent, i);
        child = v as T;
        if (child == null)
        {
            child = GetVisualChild<T>(v);
        }
        if (child != null)
        {
            break;
        }
    }
    return child;
}