WPF DataGrid 水平滚动条问题
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/14597227/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
WPF DataGrid horizontal scrollbar issue
提问by synergetic
I have the following datagrid:
我有以下数据网格:
<DataGrid x:Name="myDataGrid"
RowHeaderWidth="{Binding RelativeSource={RelativeSource Self},
Path=RowHeight}">
<DataGrid.Columns>
<DataGridTextColumn Header="Name" Width="*"
Binding="{Binding Name}"/>
<DataGridTextColumn Header="Age" Width="1.2*"
Binding="{Binding Age}"/>
</DataGrid.Columns>
</DataGrid>
<Button Grid.Row="1" Content="Add" Click="Button_Click"
Width="100"/>
private void Button_Click(object sender, RoutedEventArgs e)
{
var person = new Person()
{
Name = "Aaa",
Age = 27
};
myDataGrid.Items.Add(person);
}
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
}
The problem is horizontal scrollbar appears when a new row added, which is unnecessary. Removing RowHeaderWidth property will fix the problem, but I need this to show validation errors. Setting RowHeaderWidth to a fixed value won't help. Can someone kindly suggest me a solition?
问题是添加新行时出现水平滚动条,这是不必要的。删除 RowHeaderWidth 属性将解决问题,但我需要它来显示验证错误。将 RowHeaderWidth 设置为固定值无济于事。有人可以建议我一个solition吗?
回答by Hyde
Try this:
尝试这个:
<DataGrid x:Name="myDataGrid"
RowHeaderWidth="{Binding RelativeSource={RelativeSource Self},
Path=RowHeight}" ScrollViewer.CanContentScroll="False" ScrollViewer.HorizontalScrollBarVisibility="Disabled">
<DataGrid.Columns>
<DataGridTextColumn Header="Name" Width="*"
Binding="{Binding Name}"/>
<DataGridTextColumn Header="Age" Width="*"
Binding="{Binding Age}"/>
</DataGrid.Columns>
</DataGrid>
回答by synergetic
One workaround I found is to set width of a button, which resides at the intersection of rows and columns (left-top of a datagrid). This button appears in visual tree when a first row is added to the datagrid. I learnt about this button here.
我发现的一种解决方法是设置按钮的宽度,该按钮位于行和列的交叉处(数据网格的左上角)。当第一行添加到数据网格时,此按钮出现在可视化树中。我在这里了解了这个按钮。
public MainWindow()
{
InitializeComponent();
myDataGrid.ItemContainerGenerator.StatusChanged += onItemContainerGeneratorStatusChanged;
}
private void onItemContainerGeneratorStatusChanged(object sender, EventArgs e)
{
if (((ItemContainerGenerator)sender).Status == System.Windows.Controls.Primitives.GeneratorStatus.ContainersGenerated)
{
Button btn = GetVisualChild<Button>(myDataGrid);
if (btn != null)
{
btn.Width = myDataGrid.RowHeaderActualWidth;
}
}
}
public 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;
}
It didn't work if I set DataGrid.RowDetailsTemplate and DataGrid.SelectedItem to newly added row, though. So I tried the following:
但是,如果我将 DataGrid.RowDetailsTemplate 和 DataGrid.SelectedItem 设置为新添加的行,则它不起作用。所以我尝试了以下方法:
private void onItemContainerGeneratorStatusChanged(object sender, EventArgs e)
{
if (((ItemContainerGenerator)sender).Status == System.Windows.Controls.Primitives.GeneratorStatus.ContainersGenerated)
{
ScrollViewer sv = GetVisualChild<ScrollViewer>(myDataGrid);
if (sv != null)
{
AutomationPeer automationPeer = FrameworkElementAutomationPeer.FromElement(sv);
if (automationPeer == null)
automationPeer = FrameworkElementAutomationPeer.CreatePeerForElement(sv);
IScrollProvider provider = automationPeer.GetPattern(PatternInterface.Scroll) as IScrollProvider;
try { provider.Scroll(ScrollAmount.SmallIncrement, ScrollAmount.NoAmount); }
catch { }
try { provider.Scroll(ScrollAmount.SmallDecrement, ScrollAmount.NoAmount); }
catch { }
}
}
}
which does solve the original issue, but introduces a new one: validation error red boxes are now shifted from the text boxes, whose errors they indicate.
这确实解决了原始问题,但引入了一个新问题:验证错误红色框现在从文本框移出,它们指示其错误。
回答by synergetic
Another workaround which serves me better is:
另一个更适合我的解决方法是:
private void fixScrollBarBug()
{
ScrollBar scrollBar = GetChildByName<ScrollBar>(myDataGrid, "PART_HorizontalScrollBar");
if (scrollBar != null)
{
if (VisualTreeHelper.GetChildrenCount(scrollBar) > 0)
{
Grid grid = (Grid)VisualTreeHelper.GetChild(scrollBar, 0);
if (VisualTreeHelper.GetChildrenCount(grid) == 3)
{
try
{
RepeatButton leftButton = (RepeatButton)VisualTreeHelper.GetChild(grid, 0);
RepeatButton rightButton = (RepeatButton)VisualTreeHelper.GetChild(grid, 2);
AutomationPeer automationPeer = FrameworkElementAutomationPeer.FromElement(rightButton);
if (automationPeer == null)
automationPeer = FrameworkElementAutomationPeer.CreatePeerForElement(rightButton);
IInvokeProvider provider = automationPeer.GetPattern(PatternInterface.Invoke) as IInvokeProvider;
provider.Invoke();
automationPeer = FrameworkElementAutomationPeer.FromElement(leftButton);
if (automationPeer == null)
automationPeer = FrameworkElementAutomationPeer.CreatePeerForElement(leftButton);
provider = automationPeer.GetPattern(PatternInterface.Invoke) as IInvokeProvider;
provider.Invoke();
}
catch { }
}
}
}
}
Calling the above method after the first row added, solved the issue:
添加第一行后调用上述方法,问题解决:
myDataGrid.Dispatcher.BeginInvoke(DispatcherPriority.Input, new Action(
delegate
{
fixScrollBarBug();
}));

