WPF 不需要的网格拆分器行为

Posted

技术标签:

【中文标题】WPF 不需要的网格拆分器行为【英文标题】:WPF unwanted grid splitter behaviour 【发布时间】:2011-02-07 04:52:21 【问题描述】:

我有一个包含 3 列的简单网格(其中一个包含网格拆分器)。当调整网格的大小并且左列达到其最小宽度时,它不会什么都不做,而是增加右列的宽度。谁能帮我阻止这个?

我无法设置右列的最大宽度,因为网格本身也会调整大小。

这里有一些显示问题的示例代码。调整大小时,将鼠标移到红色区域:

XAML:

<Grid DockPanel.Dock="Top" Height="200">
    <Grid.ColumnDefinitions>
        <ColumnDefinition MinWidth="200" Width="*" />
        <ColumnDefinition Width="3" />
        <ColumnDefinition MinWidth="120" Width="240" />
    </Grid.ColumnDefinitions>
    <Grid.RowDefinitions>
        <RowDefinition Height="*" />
    </Grid.RowDefinitions>
    <Rectangle Fill="Red" Grid.Row="0" Grid.Column="0" />
    <DockPanel LastChildFill="True" Grid.Row="0" Grid.Column="2" >
        <Rectangle DockPanel.Dock="Right" Width="20" Fill="Blue" />
        <Rectangle Fill="Green" />
    </DockPanel>
    <GridSplitter Background="LightGray" Grid.Row="0" Grid.Column="1" Height="Auto" Width="Auto" HorizontalAlignment="Stretch" VerticalAlignment="Stretch" />
</Grid>

【问题讨论】:

【参考方案1】:

我知道这有点晚了,但我刚刚遇到了这个问题,这是我的解决方案。不幸的是,它不够通用,它仅适用于具有两列的网格,但它可能可以进一步适应。但是,它解决了所描述的问题和我自己的问题,所以这里是:

解决方案包含黑客或解决方法,但是您想调用它。不是为左列和右列都声明 MinWidth,而是为第一列声明 MinWidth 和 MaxWidth。这意味着 GridSplitter 不会向右移动定义的位置。到目前为止,一切都很好。

下一个问题是,如果我们有一个可调整大小的容器(在我的例子中是窗口),这还不够。这意味着我们不能随心所欲地扩大左列,即使第二列可能有足够的空间。幸运的是,有一个解决方案:绑定 Grid ActualWidth 并使用加法转换器。转换器参数实际上是右列所需的 MinWidth,显然是负值,因为我们需要从 Grid Width 中减去它。您也可以使用 SubtractConvertor,但这取决于您。

这里是 xaml 和代码:

<Grid Background="DynamicResource MainBackground" x:Name="MainGrid" >
    <Grid.ColumnDefinitions>
        <ColumnDefinition Width="200" MinWidth="100" MaxWidth="Binding Path=ActualWidth, RelativeSource=RelativeSource AncestorType=Grid, Converter=Converters:AdditionConverter, ConverterParameter=-250" />
        <ColumnDefinition Width="*" />
    </Grid.ColumnDefinitions>
    <GridSplitter  Width="3" VerticalAlignment="Stretch" Grid.Column="0"/>
    <!-- your content goes here -->
</Grid>

和转换器:

[ValueConversion(typeof(double), typeof(double))]
public class AdditionConverter : MarkupExtension, IValueConverter

    #region IValueConverter Members

    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    
        double dParameter;
        if (targetType != typeof(double) ||
            !double.TryParse((string)parameter, NumberStyles.Any, CultureInfo.InvariantCulture, out dParameter))
        
            throw new InvalidOperationException("Value and parameter passed must be of type double");
        
        var dValue = (double)value;
        return dValue + dParameter;
    

    public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    
        throw new NotSupportedException();
    

    #endregion

    #region Overrides of MarkupExtension

    /// <summary>
    /// When implemented in a derived class, returns an object that is set as the value of the target property for this markup extension. 
    /// </summary>
    /// <returns>
    /// The object value to set on the property where the extension is applied. 
    /// </returns>
    /// <param name="serviceProvider">Object that can provide services for the markup extension.
    ///                 </param>
    public override object ProvideValue(IServiceProvider serviceProvider)
    
        return this;
    

    #endregion

希望对你有帮助,

米海Drebot

【讨论】:

【参考方案2】:

这种方法是一种技巧,但可以用于达到预期的效果。尝试在 Window.SizeChanged 事件上放置一个事件处理程序来设置第一个 columndef 的 maxWidth。例如:

private void Window_SizeChanged(object sender, SizeChangedEventArgs e)

     int borderBuffer = 9;
     //              AppWindow Size    Splitter Width     Far Column Min  FudgeFactor                
     Col0.MaxWidth = e.NewSize.Width - Col1.Width.Value - Col2.MinWidth - borderBuffer;

我需要使用此方法来防止网格顶行中的第三方控件在 gridSplitter 忽略宽度设置为“自动”的列的最小/最大宽度时产生一些不希望的换行。可能需要根据您的情况调整borderBuffer。根据几何/列/边框宽度,在我的例子中,borderBuffer 并没有完全的意义——它只是对我的布局有用的幻数。

如果有人能提出更清洁的解决方案,我很乐意使用它。这个解决方案让我想起了痛苦地试图强迫 VB6 调整控件的大小 - 呸。但就目前而言,它比让我的网格顶行上的项目因意外包装而被隐藏要好。

【讨论】:

【参考方案3】:

通过将列定义中的 * 更改为 Auto 并将 240 更改为 *,我停止了这种不良行为。

【讨论】:

感谢您的回复。将列设置为 auto 不是我想要的网格行为。任何一列都应该有 *,使它填满空间。 Auto 没有这种行为。 是的,你是对的!我尝试了更多,这次我发现它是真实的,我认为...... 再次感谢,但这也不起作用 :) 我现在可以将拆分器拖出右侧。

以上是关于WPF 不需要的网格拆分器行为的主要内容,如果未能解决你的问题,请参考以下文章

如何使 wpf 行缩小以适合内容

[使用箭头键的WPF数据网格单元格循环

WPF 窗口设计器上的对齐网格

WPF扩展器单击事件上的展开/折叠网格列

如何使用一种方法在网格 C# WPF 中添加行 [重复]

WPF数据网格左栏