将一个控件的值传递给 Converter 以设置另一个控件的宽度

Posted

技术标签:

【中文标题】将一个控件的值传递给 Converter 以设置另一个控件的宽度【英文标题】:Pass the value of one control to a Converter to set the width on another control 【发布时间】:2010-09-23 15:30:32 【问题描述】:

我想根据容器的宽度减去 TextBlock 上设置的边距来设置 TextBlock 的宽度。

这是我的代码:

<TextBlock x:Name="txtStatusMessages" 
           Width="Binding ElementName=LayoutRoot,Path=ActualWidth "
                   TextWrapping="WrapWithOverflow" 
           Foreground="White" 
           Margin="5,5,5,5">This is a message
</TextBlock>

这很好用,除了 TextBlock 由于左右边距设置为 5 而太大了 10 个单位。

好的,所以我想...让我们使用转换器。但我不知道如何传递我的容器控件的 ActualWidth(见上图:LayoutRoot)。

我知道如何使用转换器,甚至是带参数的转换器,只是不是像... Binding ElementName=LayoutRoot,Path=ActualWidth 之类的参数

例如,我无法做到这一点:

Width="Binding Converter=StaticResource PositionConverter,  
       ConverterParameter=Binding ElementName=LayoutRoot,Path=ActualWidth "

我希望我说得足够清楚,并希望您能提供帮助,因为今晚 Google 对我没有帮助。

【问题讨论】:

【参考方案1】:

您应该使用另一个控件作为源,而不是参数。 参数必须是常数,在您的情况下可以是 -5。

我现在不在 VS 附近,所以语法可能不准确,但是,它类似于:

Width="Binding ElementName=LayoutRoot, Path=ActualWidth,
Converter=StaticResource PositionConverter, ConverterParameter=-5"

(转换器将接收 -5 作为字符串,在使用之前必须将其转换为数字。)

根据我的经验,最好使用 DependecyProperty XXX 的 OnXXXChanged 回调,而不是将同一窗口/根控件中的控件相互绑定。 原因之一是您可能希望稍后将它们绑定到外部元素。

或者,使用多重绑定:

<TextBlock>
    <TextBlock.Width>
        <MultiBinding Converter="StaticResource yourConverter">
            <MultiBinding.Bindings>
                <Binding /> <!-- Bind to parameter 1 here -->
                <Binding /> <!-- Bind to parameter 2 here -->
          </MultiBinding.Bindings>
        </MultiBinding>
    </TextBlock.Width>
</TextBlock>

还有一个转换器,它将两个参数转换为你想要的值。

【讨论】:

谢谢丹尼。这工作得很好。但我希望不必对参数值进行硬编码。我不知道 Parameter 需要是一个常数。谢谢! 非常感谢,太有帮助了!【参考方案2】:

是的..multi 绑定对我有用.. 实际上我尝试将元素作为转换参数发送,但它不接受。这就是我将元素作为值传递给转换器类的原因。

下面是我的例子..

<ListView ... >
<ListView.View>
<GridView>
    <GridViewColumn Header="xyz" >

        <GridViewColumn.Width>
            <MultiBinding Converter="StaticResource GetWidthfromParentControl">
                <MultiBinding.Bindings>
                    <Binding ElementName="lstNetwork" Path="ActualWidth"/>
                    <Binding ElementName="MyGridView"/>
                </MultiBinding.Bindings>
            </MultiBinding>
        </GridViewColumn.Width>
    ....
    </GridViewColumn>
    <GridViewColumn ...>
    ....
    </GridViewColumn>
</GridView>
</ListView.View>
</ListView>

在窗口调整大小时,我的第一个 gridviewcolumn 必须调整大小,而不是其他两个 gridviewcolumn.. 我通过了列表视图的实际宽度以及总的gridview对象作为一个元素.. 如果你去转换器代码...

class GetWidthfromParentControl : IMultiValueConverter

    #region IMultiValueConverter Members

    public object Convert(object[] values, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    
        GridView view = values[1] as GridView;
        GridViewColumnCollection collc = view.Columns;
        double actualWidths = collc[1].ActualWidth + collc[2].ActualWidth;
        return ((double)values[0] - actualWidths );
    

    public object[] ConvertBack(object value, Type[] targetTypes, object parameter, System.Globalization.CultureInfo culture)
    
        return null;
    

    #endregion

这对我有用... :)

【讨论】:

【参考方案3】:

虽然我怀疑可能有更好的方法来解决您的问题,但我想我对您想要做的事情有一个答案。 (您没有提到您的容器是什么类型。例如 StackPanel 会为您处理宽度计算。请参阅下面的 TextBox#2)

首先是 XAML

<Window x:Class="WpfApplication1.Window2" ...
    xmlns:local="clr-namespace:WpfApplication1"
    Title="Window2" Height="300" Width="300">
    <Window.Resources>
        <local:WidthSansMarginConverter x:Key="widthConverter" />
    </Window.Resources>
    <Grid>
        <StackPanel x:Name="stack">
            <TextBlock x:Name="txtStatusMessages" 
                    Width="Binding ElementName=stack,Path=ActualWidth, 
                        Converter=StaticResource widthConverter"
                    TextWrapping="WrapWithOverflow" 
                    Background="Aquamarine" 
                    Margin="5,5,5,5">
                This is a message
            </TextBlock>
            <TextBlock x:Name="txtWhatsWrongWithThis" 
                    TextWrapping="WrapWithOverflow" 
                    Background="Aquamarine" 
                    Margin="5,5,5,5">
                This is another message
            </TextBlock>
        </StackPanel>
    </Grid>
</Window>

接下来是转换器。我们这里有个问题..因为某种原因the ConverterParameter for the Convert methods cannot be a dynamic value。因此,我们通过在 Window 的 ctor 中设置的 Converter 的公共属性潜入 Textbox Margin。 WidthSansMarginConverter.cs

public class WidthSansMarginConverter : IValueConverter
    
        private Thickness m_Margin = new Thickness(0.0);

        public Thickness Margin
        
            get  return m_Margin; 
            set  m_Margin = value; 
        
        #region IValueConverter Members

        public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        
            if (targetType != typeof(double))  return null; 

            double dParentWidth = Double.Parse(value.ToString());
            double dAdjustedWidth = dParentWidth-m_Margin.Left-m_Margin.Right;
            return (dAdjustedWidth < 0 ? 0 : dAdjustedWidth);
        

        public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        
            throw new NotImplementedException();
        

        #endregion
    

Window2.xaml.cs

        public Window2()
        
            InitializeComponent();

            WidthSansMarginConverter obConverter = this.FindResource("widthConverter") as WidthSansMarginConverter;
            obConverter.Margin = txtStatusMessages.Margin;
        

HTH。感谢您的练习:)

【讨论】:

感谢吉舒!您的示例帮助我了解了有关转换器的更多信息。尽管您的答案有效,但我最终将 danny 的答案标记为答案,因为这是我使用的解决方案。我又累又懒,他的代码虽然不那么灵活,但代码更少。谢谢! 我一直被许多无效的解决方案所困扰,直到我找到了这个!谢谢【参考方案4】:

如果您的文本框是 LayoutRoot 的直接子级,只需在您的文本框中设置以下属性

HorizontalAlignment="Stretch"

【讨论】:

【参考方案5】:

根据http://social.msdn.microsoft.com/Forums/en-US/wpf/thread/7298ceb5-bf56-47aa-a161-5dd99189408b,如果您的转换器是从 DependencyObject 派生的,您可以将 Dependency 属性添加到您的自定义转换器。

在这种情况下,您甚至可以使用数据绑定将值传递给您在 XAML 中定义转换器(在资源字典中)的那些属性。

【讨论】:

以上是关于将一个控件的值传递给 Converter 以设置另一个控件的宽度的主要内容,如果未能解决你的问题,请参考以下文章

将 controlName 传递给另一个数组时,类型 any[] 上不存在属性控件名称?

将参数从jsp动态传递给另一个jsp

将变量传递给另一个页面上的函数

如何使用html将一个页面的值传递给另一个页面

获取传递给完成处理程序以进行单元测试的值

REACTJS 和 API 将 Api 的值传递给另一个页面