WPF全新,对WinForms非常熟悉(可能会使转换变得更加粗糙)。我正在尝试将旧的WinForms项目中的一些功能移植到WPF作为学习体验。
目标是在DataGrid中查找与TextBox中的字符串匹配的单元格值。我找到了一个great example使用绑定来完成它。基本上链接的代码会将任何匹配的DataGridCell的背景颜色更改为橙色。我已经修改了我的版本,但功能应该是相同的。请参阅代码示例的链接,这里提供它似乎有点多余。填充我的DataGrid的数据来自DataTable(如果重要的话)。
我想要做的是有一个“下一步”按钮,它将遍历每个单元格(通过使用背景颜色或自定义属性DataGridTextSearch.IsTextMatch确定)并选择它。似乎可以只修改一些提供的代码,但我不知道从哪里开始。在我的旧WinForms项目中,我将DataGridViewCell存储在一个列表中(在使用Linq查询找到它们之后)并且只是附加了按钮行为以递增所述列表并设置当前单元格。我怀疑可能有更聪明/更好的方式涉及绑定,我甚至不知道如何将这些匹配的单元格添加到列表中,如果这是一个选项。
因此,总而言之,我想要一个循环遍历特定DataGridCells的按钮(基于Background或自定义DataGridTextSearch.IsTextMatch属性)并选择它们。
提前致谢。
答案 0 :(得分:5)
根据您在问题中提供的link,我已经找到了解决方法。在我的解决方案中,当DataGridCell
与TextBox
中的字符串匹配时,它的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;
}