验证WPF文本框中的十进制数字

本文关键字:十进制数字 WPF 文本 验证 | 更新日期: 2023-09-27 18:25:57

我想在WPF TextBoxes上实现十进制数验证。

我找到了WPF TextBox对数值验证的答案,但我想要一个允许十进制值(浮点/双精度)的TextBox。

我该怎么做?

public static bool GetIsDecimal(DependencyObject obj)
    {
        return (bool)obj.GetValue(IsDecimalProperty);
    }
    public static void SetIsDecimal(DependencyObject obj, bool value)
    {
        obj.SetValue(IsDecimalProperty, value);
    }
    public static readonly DependencyProperty IsDecimalProperty =
 DependencyProperty.RegisterAttached("IsDecimal", typeof(bool), typeof(TextBoxHelpers), new PropertyMetadata(false, new PropertyChangedCallback((s, e) =>
 {
     TextBox targetTextbox = s as TextBox;
     if (targetTextbox != null)
     {
         if ((bool)e.OldValue && !((bool)e.NewValue))
         {
             targetTextbox.PreviewTextInput -= targetTextbox_PreviewTextInput;
         }
         if ((bool)e.NewValue)
         {
             targetTextbox.PreviewTextInput += targetTextbox_PreviewTextInput;
             targetTextbox.PreviewKeyDown += targetTextbox_PreviewKeyDown;
         }
     }
 })));
    static void targetTextbox_PreviewKeyDown(object sender, KeyEventArgs e)
    {
        e.Handled = (e.Key == Key.Space);
    }
    static void targetTextbox_PreviewTextInput(object sender, TextCompositionEventArgs e)
    {
        Char newChar = e.Text.ToString()[0];
        e.Handled = !(Char.IsNumber(newChar) || (newChar == '.'));
    }
}

我正在使用上面的代码。代码(Char.IsNumber(newChar) || (newChar == '.'))将检查数字和小数。所以现在TextBox只允许数字和decimal(.)。但问题是,我可以输入多个小数点(例如,1.01.22.011)。所以我想限制输入多个十进制点。

验证WPF文本框中的十进制数字

您可以使用行为

    <TextBox MaxLength="17" MinWidth="205" HorizontalContentAlignment="Right"
             Text="{Binding Path=Amount, Mode=TwoWay, UpdateSourceTrigger=LostFocus, ValidatesOnExceptions=true,  ValidatesOnDataErrors=true, NotifyOnValidationError=True,
                             TargetNullValue={x:Static System:String.Empty},
                             Converter={StaticResource MyStringToDecimalConverter},ConverterParameter=#',##0.00}">            
        <i:Interaction.Behaviors>                
            <Behaviors:TextBoxInputBehavior InputMode="DecimalInput" JustPositivDecimalInput="false"//>
        </i:Interaction.Behaviors>
    </TextBox>

行为.cs:

public class TextBoxInputBehavior : Behavior<TextBox>
{
    const NumberStyles validNumberStyles = NumberStyles.AllowDecimalPoint |
                                               NumberStyles.AllowThousands |
                                               NumberStyles.AllowLeadingSign;
    public TextBoxInputBehavior()
    {
        this.InputMode = TextBoxInputMode.None;
        this.JustPositivDecimalInput = false;
    }
    public TextBoxInputMode InputMode { get; set; }

    public static readonly DependencyProperty JustPositivDecimalInputProperty =
     DependencyProperty.Register("JustPositivDecimalInput", typeof(bool),
     typeof(TextBoxInputBehavior), new FrameworkPropertyMetadata(false));
    public bool JustPositivDecimalInput
    {
        get { return (bool)GetValue(JustPositivDecimalInputProperty); }
        set { SetValue(JustPositivDecimalInputProperty, value); }
    }
    protected override void OnAttached()
    {
        base.OnAttached();
        AssociatedObject.PreviewTextInput += AssociatedObjectPreviewTextInput;
        AssociatedObject.PreviewKeyDown += AssociatedObjectPreviewKeyDown;
        DataObject.AddPastingHandler(AssociatedObject, Pasting);
    }
    protected override void OnDetaching()
    {
        base.OnDetaching();
        AssociatedObject.PreviewTextInput -= AssociatedObjectPreviewTextInput;
        AssociatedObject.PreviewKeyDown -= AssociatedObjectPreviewKeyDown;
        DataObject.RemovePastingHandler(AssociatedObject, Pasting);
    }
    private void Pasting(object sender, DataObjectPastingEventArgs e)
    {
        if (e.DataObject.GetDataPresent(typeof(string)))
        {
            var pastedText = (string)e.DataObject.GetData(typeof(string));
            if (!this.IsValidInput(this.GetText(pastedText)))
            {
                System.Media.SystemSounds.Beep.Play();
                e.CancelCommand();
            }
        }
        else
        {
            System.Media.SystemSounds.Beep.Play();
            e.CancelCommand();
        }
    }
    private void AssociatedObjectPreviewKeyDown(object sender, KeyEventArgs e)
    {
        if (e.Key == Key.Space)
        {
            if (!this.IsValidInput(this.GetText(" ")))
            {
                System.Media.SystemSounds.Beep.Play();
                e.Handled = true;
            }
        }
    }
    private void AssociatedObjectPreviewTextInput(object sender, TextCompositionEventArgs e)
    {
        if (!this.IsValidInput(this.GetText(e.Text)))
        {
            System.Media.SystemSounds.Beep.Play();
            e.Handled = true;
        }
    }
    private string GetText(string input)
    {
        var txt = this.AssociatedObject;
        var realtext = txt.Text.Remove(txt.SelectionStart, txt.SelectionLength);
        var newtext = realtext.Insert(txt.CaretIndex, input);
        return newtext;
    }
    private bool IsValidInput(string input)
    {
        switch (InputMode)
        {
            case TextBoxInputMode.None:
                return true;
            case TextBoxInputMode.DigitInput:
                return CheckIsDigit(input);
            case TextBoxInputMode.DecimalInput:
                decimal d;
                //wen mehr als ein Komma
                if (input.ToCharArray().Where(x => x == ',').Count() > 1)
                    return false;
                if (input.Contains("-"))
                {
                    //minus einmal am anfang zulässig
                    if (!this.JustPositivDecimalInput && input.IndexOf("-") == 0  && input.Length == 1)
                        return true;
                    else
                    {
                        var result = decimal.TryParse(input, validNumberStyles, CultureInfo.CurrentCulture, out d);
                        return result;
                    }
                }
                else
                {
                    var result = decimal.TryParse(input, validNumberStyles, CultureInfo.CurrentCulture, out d);
                    return result;
                }

            default: throw new ArgumentException("Unknown TextBoxInputMode");
        }
        return true;
    }
    private bool CheckIsDigit(string wert)
    {
        return wert.ToCharArray().All(Char.IsDigit);
    }
}
public enum TextBoxInputMode
{
    None,
    DecimalInput,
    DigitInput
}
<TextBox  Height="36"   PreviewTextInput="NumericTextBoxInput"   Width="235"/>

C#代码:

 void NumericTextBoxInput(object sender, TextCompositionEventArgs e)
    {
        var regex = new Regex(@"^[0-9]*(?:'.[0-9]*)?$");
        if (regex.IsMatch(e.Text) && !(e.Text == "." && ((TextBox)sender).Text.Contains(e.Text)))
               e.Handled = false;
        else
            e.Handled = true;
    }
<TextBox  Height="36"   PreviewTextInput="NumberDecimalValidationTextbox"   Width="235"/>

  private void NumberDecimalValidationTextbox(object sender, TextCompositionEventArgs e)
        {
            var regex = new Regex(@"^[0-9]*(?:'.[0-9]*)?$");
            if (regex.IsMatch(e.Text) && !(e.Text == "." && ((TextBox)sender).Text.Contains(e.Text)))
                e.Handled = false;
            else
                e.Handled = true;
        }

我使用的"."验证了文本框中的浮点输入

void NumericTextBoxInput(object sender, TextCompositionEventArgs e)
    {
        var regex = new Regex(@"^[0-9]*(?:'.[0-9]*)?$");
        if (regex.IsMatch(e.Text) && !(e.Text == "." && ((TextBox)sender).Text.Contains(e.Text)))
               e.Handled = false;
        else
            e.Handled = true;
    }

现在我试着做同样的事情,包括一个","作为十进制标记。下面的regex是可以的,但我不允许只使用一个"."或一个","。

var regex = new Regex(@"^[0-9]*(?:('.|',)[0-9]*)?$");

有人知道如何只允许"、"或"."出现一次吗?非常感谢