调整包含包装TextBlocks的DataGrid的问题

时间:2012-09-10 11:25:13

标签: c# wpf layout datagrid

我对.NET 4.0 DataGrid有一个非常令人不安的问题。我有比例模板列包含TextBlock并启用了textWrapping。

问题是,DataGrid的高度在加载时是不正确的(它的大小就好像文本块都以最大值包装一样。)并且在调整大小时不会更新它们的大小。它似乎是一个布局问题(当比例大小未解析时,似乎会调用MeasureOverride和ArrangeOverride,之后不会调用它......)但我无法解决它。

以下是显示问题的简化代码:

MainWindow.xaml

<Window x:Class="WpfApplication1.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="MainWindow" Height="700" Width="525">
<StackPanel Width="500">
    <Button Content="Add DataGrid" Click="Button_Click" />
    <ItemsControl x:Name="itemsControl">
        <ItemsControl.ItemContainerStyle>
            <Style>
                <Setter Property="Control.Margin" Value="5" />
            </Style>
        </ItemsControl.ItemContainerStyle>
    </ItemsControl>
</StackPanel>

MainWindow.xaml.cs

using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Media;

namespace WpfApplication1
{
public partial class MainWindow : System.Windows.Window
{
    public MainWindow()
    {
        InitializeComponent();
    }

    private void Button_Click(object sender, RoutedEventArgs e)
    {
        itemsControl.Items.Add(CreateDataGrid()); 
    }

    private DataGrid CreateDataGrid()
    {
        var dataGrid = new DataGrid() { HeadersVisibility = DataGridHeadersVisibility.Column };
        dataGrid.MaxWidth = 500;
        dataGrid.Background = Brushes.LightSteelBlue;
        dataGrid.Columns.Add(GetDataGridTemplateColumn("Label", "Auto"));
        dataGrid.Columns.Add(GetDataGridTemplateColumn("Value", "*"));
        dataGrid.Items.Add(new Entry() { Label = "Text Example 1", Value = "Some wrapped text" });
        dataGrid.Items.Add(new Entry() { Label = "Text Example 2", Value = "Some wrapped text" });
        return dataGrid;
    }

    private DataGridTemplateColumn GetDataGridTemplateColumn(string bindingPath, string columnWidth)
    {
        DataGridTemplateColumn result = new DataGridTemplateColumn() { Width = (DataGridLength)(converter.ConvertFrom(columnWidth))};
        FrameworkElementFactory cellTemplateframeworkElementFactory = new FrameworkElementFactory(typeof(TextBox));
        cellTemplateframeworkElementFactory.SetValue(TextBox.NameProperty, "cellContentControl");
        cellTemplateframeworkElementFactory.SetValue(TextBox.TextProperty, new Binding(bindingPath));
        cellTemplateframeworkElementFactory.SetValue(TextBox.TextWrappingProperty, TextWrapping.Wrap);
        result.CellTemplate = new DataTemplate() { VisualTree = cellTemplateframeworkElementFactory };
        return result;
    }

    private static DataGridLengthConverter converter = new DataGridLengthConverter();
}

public class Entry
{
    public string Label { get; set; }
    public string Value { get; set; }
}

}

enter image description here

enter image description here

2 个答案:

答案 0 :(得分:0)

我终于提出了一个相当肮脏的解决方案,但至少它是有效的:我通过计算自己正确的值手动更新网格的高度,考虑到行高,网格填充,网格边框和ColumnHeaderRow高度。我需要在OnPropertyChanged(用于填充和网格边框),DataGridColumn.SizeChanged,DataGridColumn.RowUnloaded,ColumnHeaderPresenter.SizeChanged和其他一些更新它。

唯一的问题是它可以正常使用默认的DataGrid ControlTemplate,但如果模板要更改网格渲染则不再正确。

答案 1 :(得分:0)

最后找到了一个解决方案:在DataGrid上将CanContentScroll设置为false修复了这个问题。

<Setter Property="ScrollViewer.CanContentScroll" Value="False" />