在XAML中设置TextBox的宽度,以便它能够显示x位数

时间:2015-06-05 02:09:54

标签: wpf xaml

我想设置WPF TextBox的宽度,以便它有足够的空间,例如,任何5位数的TCP端口号。它不应该更大,它不应该动态调整大小,即Width =" Auto"不是我想要的。

我正在寻找一种通用的方式,即尊重所用字体的方式,而且我不想在字体 - 或其他任何东西时必须使用像素精确的宽度值。可能会改变5位数的像素宽度 - 已更改。

我想通过MeasureString可以在代码中执行 - 如果不方便 - 但在XAML中可以使用吗?

1 个答案:

答案 0 :(得分:2)

嗯,这可能不完美,但这是一个可能的解决方案。

创建一个ControlTemplate,其中包含所需的CharacterLengthGhostString依赖项属性。

public class DynamicTextBox : TextBox
{
    public int CharacterLength
    {
        get { return (int)GetValue(CharacterLengthProperty); }
        set { SetValue(CharacterLengthProperty, value); }
    }

    // Using a DependencyProperty as the backing store for CharacterLength.  This enables animation, styling, binding, etc...
    public static readonly DependencyProperty CharacterLengthProperty =
        DependencyProperty.Register("CharacterLength", typeof(int), typeof(DynamicTextBox), new PropertyMetadata(5, new PropertyChangedCallback(CharacterLengthChanged)));

    public string GhostString
    {
        get { return (string)GetValue(GhostStringProperty); }
        private set { SetValue(GhostStringProperty, value); }
    }

    // Using a DependencyProperty as the backing store for GhostString.  This enables animation, styling, binding, etc...
    public static readonly DependencyProperty GhostStringProperty =
        DependencyProperty.Register("GhostString", typeof(string), typeof(DynamicTextBox), new PropertyMetadata("#####"));

    static DynamicTextBox()
    {
        DefaultStyleKeyProperty.OverrideMetadata(typeof(DynamicTextBox), new FrameworkPropertyMetadata(typeof(DynamicTextBox)));
    }

    private static void CharacterLengthChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        DynamicTextBox textbox = d as DynamicTextBox;

        string ghost = string.Empty;

        for (int i = 0; i < textbox.CharacterLength; i++)
            ghost += "#";

        textbox.GhostString = ghost;
    }
}

每当CharacterLength属性发生变化时,GhostString属性都将重新计算,您将在一分钟内看到魔法。

为此新控件设置StyleControlTemplate

<Style TargetType="{x:Type local:DynamicTextBox}"
       BasedOn="{StaticResource {x:Type TextBox}}">
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type local:DynamicTextBox}">
                <Border Background="{TemplateBinding Background}"
                        BorderBrush="{TemplateBinding BorderBrush}"
                        BorderThickness="{TemplateBinding BorderThickness}">
                    <Grid>
                        <TextBlock Text="{TemplateBinding GhostString}"
                                   Visibility="Hidden"
                                   Margin="3,0"/>

                        <ScrollViewer Margin="0"
                                  x:Name="PART_ContentHost" />
                    </Grid>
                </Border>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
</Style>

GhostString属性位于隐藏 TextBlock内,这意味着渲染宽度,但文字不可见,它放在无论如何TextBox

您可以像这样使用控件:

<Controls:DynamicTextBox CharacterLength="12" HorizontalAlignment="Left"/>
    <Controls:DynamicTextBox CharacterLength="6" HorizontalAlignment="Left"/>
    <Controls:DynamicTextBox CharacterLength="2" HorizontalAlignment="Left"/>

注意:我将Horizo​​ntalAlignment放在那里只是为了强制宽度崩溃。

结果如下:

TextBoxes inside a StackPanel

它并不完美,但它肯定是一个开始。如果你想进一步限制TextBox的宽度,我相当肯定你可以在ControlTemplate内做一些聪明的约束。