WPF TextBox 输入限制 详解

总目录


前言

通常对于WPF输入框 输入的控制无非以下方式

  • 1 直接禁止输入(包括粘贴) 不符合要求的字符
    • 如只可输入数字的输入框,当你输入字母的时候是无法输入的
  • 2 输入后,校验内容是否符合要求,然后提示错误,禁止提交信息
    • 如只可输入手机号的输入框,你输入了不符合要求的格式,会提示格式有误,提交信息的时候再次提示数据格式有误,纠正后提交
    • 该方式一般通过ErrorTemplate来实现错误信息的展示,再配合IDataErrorInfo+DataAnnotations 完成对数据的验证
  • 3 使用方式1和方式2 配合完成输入限制

本文主要针对第一种输入控制方式进行介绍。


一、预备内容

1. 预备知识

  • 熟悉 C# 和 WPF
  • 大部分的输入控制都是基于正则表达式对于输入内容进行验证的,因此有兴趣的可以先阅读 C# 正则表达式 详解 一文进行了解。

2. 准备代码

  • 新建一个WPF项目
  • MainWindow.xaml 代码如下:
<Window x:Class="WpfApp1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local="clr-namespace:WpfApp1"
        mc:Ignorable="d"
        Title="MainWindow" Height="450" Width="800">
    <Grid Background="LightGray">
        <TextBox Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46"></TextBox>
    </Grid>
</Window>

在这里插入图片描述
此时输入框,没有输入的限制,可以在输入框内随意输入我们想要输入的内容。

二、通过TextBox的相关事件实现输入限制

1. 通过 PreviewTextInput 事件

Occurs when this element gets text in a device-independent manner.
当此元素以 与设备(器件)无关的的方式 获取文本时发生。
或理解为 当输入框文本输入时发生。

1)实现只可输入数字

<TextBox Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
         PreviewTextInput="TextBox_PreviewTextInput">
</TextBox>
        private void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);
        }
  • e.Text 表示的是每次输入的单个字符,而非输入框内的整个内容
  • 当输入的Text 符合 正则表达式 则返回true
  • e.Handled = true 表示 不可输入,反之 e.Handled = false 表示可输入
  • e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);表示,只可输入0-9的数字

2)解决输入法的问题

在这里插入图片描述

问题: 上面的案例中,当输入法为 英语模式的时候,输入框可以控制,只可输入数字;
但是当我们切换到中文模式,输入框在中文模式 下 还是可以输入 其他字符,如何解决呢?
很简单,给TextBox 设置 InputMethod.IsInputMethodEnabled="False"

        <TextBox Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False"
                 PreviewTextInput="TextBox_PreviewTextInput">
        </TextBox>

通过设置 InputMethod.IsInputMethodEnabled="False" 禁用了输入法,那么当焦点在该输入框的时候,就会切换回 英语模式并且禁止切换输入模式。

3)解决粘贴的问题

问题: 上面的案例中,已经基本解决了我们的需求,但是还存在一个细节问题,如果我们需要从别的地方复制了 包含字母和中文的文本信息,我们发现可以粘贴进 TextBox 内,这就造成了输入内容不符合要求。

  • 解决方案:取消 TextBox所有的粘贴事件
<TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
         InputMethod.IsInputMethodEnabled="False"
         PreviewTextInput="TextBox_PreviewTextInput">
</TextBox>
        public MainWindow()
        {
            InitializeComponent();
            DataObject.AddPastingHandler(this.tb, TextBoxPastingEvent);
        }

        private void TextBoxPastingEvent(object sender, DataObjectPastingEventArgs e)
        {
        	//取消 粘贴命令
            e.CancelCommand();
        }
        
        private void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);
        }
  • 解决方案2:扩展,选择性粘贴
    • 通过 e.DataObject.GetData(typeof(string)) as string; 获取得到复制的文本
    • 然后可以根据具体的需求,对复制的文本 做进一步的限制
    • 如下案例实现:当复制的内容不符合正则要求的时候,取消粘贴,反之,符合要求,可以粘贴
        private void TextBoxPastingEvent(object sender, DataObjectPastingEventArgs e)
        {
            //string nums = "0123456789";
            string clipboard = e.DataObject.GetData(typeof(string)) as string;
            foreach (var item in clipboard)
            {
                //if (nums.Contains(item))
                if (new Regex("[^0-9]+").IsMatch(item.ToString()))
                {
                    e.CancelCommand();
                    break;
                }
            }
        }

4)完整实现只可输入数字

完整的案例代码如下:

        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False"
                 PreviewTextInput="TextBox_PreviewTextInput">
        </TextBox>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            DataObject.AddPastingHandler(this.tb, TextBoxPastingEvent);
        }

        private void TextBoxPastingEvent(object sender, DataObjectPastingEventArgs e)
        {
            //string nums = "0123456789";
            string clipboard = e.DataObject.GetData(typeof(string)) as string;
            foreach (var item in clipboard)
            {
                //if (nums.Contains(item))
                if (new Regex("[^0-9]+").IsMatch(item.ToString()))
                {
                    e.CancelCommand();
                    break;
                }
            }
        }

        private void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);
        }
    }

2. 通过PreviewKeyDownKeyDown事件

通过PreviewKeyDownKeyDown事件实现输入限制,主要是针对KeyModifierKeys的处理。

PreviewKeyDown 表示当焦点在该元素上时按下某个键后发生,

  • Key是C#下的枚举类型,枚举了所有的键盘键
    • 如D0到D9为0到9的按键, Key.Delete代表的是删除键, Key.OemPeriod为小数点。
  • ModifierKeys也是C#下的枚举类型,包括Alt、Ctrl、Shift等键。表示组合键,如Ctrl+A 中的Ctrl
// 指定修改键集
public enum ModifierKeys
{
    // 摘要: No modifiers are pressed.
    None = 0,
    // 摘要: The ALT key.
    Alt = 1,
    // 摘要: The CTRL key.
    Control = 2,
    // 摘要: The SHIFT key.
    Shift = 4,
    // 摘要: The Windows logo key.
    Windows = 8
}

1)实现只可输入数字

TextBox通过PreviewKeyDown捕获按下的键值,然后判断按下的键值是否符合要求,符合要求的可以键入,反之不可以键入。

具体实现代码如下:

        public static bool IsInputNumber(KeyEventArgs e)
        {
            //表示可以使用Ctrl+C Ctrl+V 等等组合键
            //如果禁止使用组合键,也可以注释这部分代码
            if (e.KeyboardDevice.Modifiers != ModifierKeys.None)
            {
                return true;
            }
            // 表示可以键入 数字0-9 以及常用的Delete,Back等辅助按键
            // 这里后续也可根据具体需求去调整
            if ((e.Key >= Key.D0 && e.Key <= Key.D9)
                || (e.Key >= Key.NumPad0 && e.Key <= Key.NumPad9)
                || e.Key == Key.Delete 
                || e.Key == Key.Back 
                || e.Key == Key.Left 
                || e.Key == Key.Right)
            {
                return true;
            }
            return false;
        }
        <TextBox Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False"
                 PreviewKeyDown="TextBox_PreviewKeyDown">
        </TextBox>

注意:这里也需要设置InputMethod.IsInputMethodEnabled="False"

        private void TextBox_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            e.Handled =!IsInputNumber(e);
        }

2)解决粘贴的问题

  • 方式1:使用第一小节中DataObject.AddPastingHandler 的方法实现
  • 方式2:继续完善 PreviewKeyDown 事件的键入逻辑
        public static bool IsInputNumber(KeyEventArgs e)
        {
            //表示可以使用Ctrl+C Ctrl+V 等等组合键
            //如果禁止使用组合键,也可以注释这部分代码
            //if (e.KeyboardDevice.Modifiers != ModifierKeys.None&& e.KeyStates != Keyboard.GetKeyStates(Key.V))
            if (e.KeyboardDevice.Modifiers != ModifierKeys.None && e.Key != Key.V)
            {
                return true;
            }

            // 表示可以键入 数字0-9 以及常用的Delete,Back等辅助按键
            // 这里后续也可根据具体需求去调整
            if ((e.Key >= Key.D0 && e.Key <= Key.D9)
                || (e.Key >= Key.NumPad0 && e.Key <= Key.NumPad9)
                || e.Key == Key.Delete
                || e.Key == Key.Back
                || e.Key == Key.Left
                || e.Key == Key.Right)
            {
                return true;
            }

            return false;
        }
  • if (e.KeyboardDevice.Modifiers != ModifierKeys.None && e.Key != Key.V)
    • 表示只要组合键中不含V 的 都可以使用,那么Ctrl+V 就不可以使用

3)解决输入框右键菜单的问题

在这里插入图片描述

在上面的案例中虽然禁止了键盘上粘贴的问题,但是如果使用鼠标右键菜单,然后进行粘贴,还是会出现问题,因此我们需要对这部分进行处理。

处理这个问题也很简单,TextBox上设置 ContextMenu="{x:Null}" 即可

4)完整实现只可输入数字

        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False" 
                 ContextMenu="{x:Null}"
                 PreviewKeyDown="TextBox_PreviewKeyDown">
        </TextBox>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private void TextBox_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            e.Handled = !IsInputNumber(e);
        }

        public static bool IsInputNumber(KeyEventArgs e)
        {
            //表示可以使用Ctrl+C Ctrl+V 等等组合键
            //如果禁止使用组合键,也可以注释这部分代码
            //if (e.KeyboardDevice.Modifiers != ModifierKeys.None&& e.KeyStates != Keyboard.GetKeyStates(Key.V))
            if (e.KeyboardDevice.Modifiers != ModifierKeys.None && e.Key != Key.V)
            {
                return true;
            }

            // 表示可以键入 数字0-9 以及常用的Delete,Back等辅助按键
            // 这里后续也可根据具体需求去调整
            if ((e.Key >= Key.D0 && e.Key <= Key.D9)
                || (e.Key >= Key.NumPad0 && e.Key <= Key.NumPad9)
                || e.Key == Key.Delete
                || e.Key == Key.Back
                || e.Key == Key.Left
                || e.Key == Key.Right)
            {
                return true;
            }

            return false;
        }
    }

3. TextChanged 事件

TextChanged 事件:在文本元素中的内容更改的情况下发生。

通过以上的PreviewTextInput 等事件,可以很好的控制输入,但是如果TextBox的Text属性,并不是通过鼠标键盘等外在设备输入的,而是通过业务逻辑直接设置的呢?

如: 在某个按钮点击事件中设置this.textBox.Text = "hello 123!";,那么就无法触发PreviewTextInputPreviewKeyDownKeyDown事件,这个时候如果需要检测设置的Text是否符合要求,就需要通过TextChanged 事件 去完成。

如: 当使用 Backspace 按键,删除TextBox中Text 的时候,可以触发PreviewKeyDownKeyDown事件,但是无法触发PreviewTextInput 事件,这个时候如果需要检测设置的Text是否符合要求,通过TextChanged 事件 去完成就比较合理。

案例:点击按钮随机文本信息并复制给TextBox的Text属性,然后通过TextChanged对不符合要求的文本信息进行处理

    <Grid Background="LightGray">
        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False" 
                 ContextMenu="{x:Null}" TextChanged="tb_TextChanged">
        </TextBox>
        <Button Click="Button_Click" Height="50" Width="200" Tag="" VerticalAlignment="Bottom" Background="Red" Content="test"></Button>
    </Grid>
        private void tb_TextChanged(object sender, TextChangedEventArgs e)
        {
            string text = this.tb.Text;
            if (!double.TryParse(text, out var result))
            {
            	// 假设这里的文本信息需要可转为double 类型的数据
            	// 直接将不符合要求的信息,显示为空
                this.tb.Text = string.Empty;
            }
        }

        private void Button_Click(object sender, RoutedEventArgs e)
        {
            string[] values = { "Hello 123", "12123", "daw", "1200" };
            this.tb.Text = values[new Random().Next(values.Length)];
        }

我们可以通过TextChanged 检测TextBox的Text 属性变化,从而做出相关的业务处理。

4. 多个事件配合使用

由以上的案例我们可知,不同的事件侧重点是不一样的,如:

  • PreviewTextInput 事件 用于检测输入框内每次输入的单个字符,从而对输入的字符进行控制
  • PreviewKeyDown事件 用于检测在输入框内所有按下的按键,包括Alt Ctrl 等按键,从而对按下的键进行控制
  • TextChanged 事件 用于检测输入框内文本信息的变化,只要变化了就会触发该事件,从而对文本信息进行处理和控制

通过以上案例还应该知道,对于某些复杂的输入限制,是需要我们多个事件配合使用的,而非仅仅使用某一个事件就能够完成,因此我们还是需要分别了解这些事件对应的特点,这样才有利于我们做好输入控制。

配合使用案例如下:
在这里插入图片描述

    <Grid Background="LightGray">
        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False"
                 PreviewTextInput="TextBox_PreviewTextInput"
                 PreviewKeyDown="tb_PreviewKeyDown">
        </TextBox>
        <Button Click="Button_Click" Height="50" Width="200" Tag="" VerticalAlignment="Bottom" Background="Red" Content="test"></Button>
    </Grid>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            DataObject.AddPastingHandler(this.tb, TextBoxPastingEvent);
        }
        private void TextBoxPastingEvent(object sender, DataObjectPastingEventArgs e)
        {
            //string nums = "0123456789";
            string clipboard = e.DataObject.GetData(typeof(string)) as string;
            foreach (var item in clipboard)
            {
                //if (nums.Contains(item))
                if (new Regex("[^0-9]+").IsMatch(item.ToString()))
                {
                    e.CancelCommand();
                    break;
                }
            }
        }

        private void tb_TextChanged(object sender, TextChangedEventArgs e)
        {
            string text = this.tb.Text;
            if (!double.TryParse(text, out var result))
            {
                this.tb.Text = string.Empty;
            }
        }

        private void Button_Click(object sender, RoutedEventArgs e)
        {
            string[] values = { "Hello 123", "12123", "daw", "1200" };
            this.tb.Text = values[new Random().Next(values.Length)];
        }

        private void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);
        }

        private void tb_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            //禁止使用 空格键
            if (e.Key == Key.Space)
            {
                e.Handled = true;
            }
        }
    }

在该案例中:

  • PreviewTextInput 负责校验输入的字符,排除不符合要求的字符
  • PreviewKeyDown 负责校验按下的按键,排除了空格键
  • TextChanged 负责检测文本的变化
  • 通过多个事件组合使用,完整详细的实现了对应的输入限制需求

三、通过附加属性实现输入限制

  • (本人)推荐这种方式,灵活性高,扩展性强,对业务代码的侵入性更小
  • 假如我们需要新增一类输入控制 或者灵活配置输入控制
    • 我们只需新增附加属性,然后在TextBox上设置附加属性的值即可完成输入框的输入限制
    • 相较于使用事件去实现输入控制,就会需要在相关事件中代码中写相关的代码,如果另一个地方需要使用输入限制,还需要重新写一遍,这种方式灵活性和可扩展性并不高,代码也得不到复用。
    • 其实这种方式本质上也是基于TextBox的各种实现实现的,不过封装的更好

1. 实现只可输入数字

    public class TextBoxAttachedProperties
    {
        public static bool GetIsOnlyNumber(DependencyObject obj)
        {
            return (bool)obj.GetValue(IsOnlyNumberProperty);
        }

        public static void SetIsOnlyNumber(DependencyObject obj, bool value)
        {
            obj.SetValue(IsOnlyNumberProperty, value);
        }

        public static readonly DependencyProperty IsOnlyNumberProperty = DependencyProperty.RegisterAttached(
                "IsOnlyNumber",
                typeof(bool),
                typeof(TextBoxAttachedProperties),
                new PropertyMetadata(false, OnIsOnlyNumberChanged));

        private static void OnIsOnlyNumberChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            if (d is TextBox textBox)
            {
                textBox.SetValue(InputMethod.IsInputMethodEnabledProperty, false);
                //textBox.SetValue(TextBox.ContextMenuProperty,null);
                bool value = (bool)e.NewValue;
                if (value)
                {
                    textBox.PreviewTextInput += TextBox_PreviewTextInput;
                    textBox.PreviewKeyDown += TextBox_PreviewKeyDown;
                    DataObject.AddPastingHandler(textBox, TextBox_PastingEvent);
                }
                else
                {
                    textBox.PreviewTextInput -= TextBox_PreviewTextInput;
                    textBox.PreviewKeyDown -= TextBox_PreviewKeyDown;
                    DataObject.RemovePastingHandler(textBox, TextBox_PastingEvent);
                }
            }
        }

        private static void TextBox_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            //禁止使用 空格键
            if (e.Key == Key.Space)
            {
                e.Handled = true;
            }
        }

        private static void TextBox_PastingEvent(object sender, DataObjectPastingEventArgs e)
        {
            //取消 粘贴命令
            e.CancelCommand();
        }

        private static void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            e.Handled = new Regex("[^0-9]+").IsMatch(e.Text);
        }
    }
 
 

上面的代码中主要的业务逻辑:

  • 1 在实例化PropertyMetadata时,通过附加属性的 依赖对象获取得到 TextBox
  • 2 再通过TextBox 使用SetValue相关属性值
  • 3 再通过TextBox 注册PreviewTextInput 等事件对输入内容进行限制

使用附加属性:

        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 local:TextBoxAttachedProperties.IsOnlyNumber="True">
        </TextBox>

2. 其他输入限制

通过上面的案例,我们就了解了使用附加属性实现输入限制了的基本逻辑。
有了基本逻辑,遇到相关需求,只需根据实际业务需求做出相关调整即可。

  • 扩展:通过正则表达式实现通用的输入限制
    public class TextBoxAttachedProperties
    {
        public static string GetRegexPattern(DependencyObject obj)
        {
            return (string)obj.GetValue(RegexPatternProperty);
        }

        public static void SetRegexPattern(DependencyObject obj, string value)
        {
            obj.SetValue(RegexPatternProperty, value);
        }

        public static readonly DependencyProperty RegexPatternProperty = DependencyProperty.RegisterAttached(
                "RegexPattern",
                typeof(string),
                typeof(TextBoxAttachedProperties),
                new PropertyMetadata(string.Empty, OnRegexPatternChanged));

        private static void OnRegexPatternChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            if (d is TextBox textBox)
            {
                string value = (string)e.NewValue;
                if (!string.IsNullOrEmpty(value))
                {
                    textBox.PreviewTextInput += TextBox_PreviewTextInput;
                    textBox.PreviewKeyDown += TextBox_PreviewKeyDown;
                    textBox.TextChanged += TextBox_TextChanged;
                    DataObject.AddPastingHandler(textBox, TextBox_PastingEvent);
                }
                else
                {
                    textBox.PreviewTextInput -= TextBox_PreviewTextInput;
                    textBox.PreviewKeyDown -= TextBox_PreviewKeyDown;
                    textBox.TextChanged -= TextBox_TextChanged;
                    DataObject.RemovePastingHandler(textBox, TextBox_PastingEvent);
                }
            }
        }

        private static void TextBox_TextChanged(object sender, TextChangedEventArgs e)
        {
            //TODO 相关业务处理
            // ...
        }

        private static void TextBox_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            //TODO 相关业务处理
            // ...
           
            //禁止使用 空格键
            //if (e.Key == Key.Space)
            //{
            //    e.Handled = true;
            //}
        }

        private static void TextBox_PastingEvent(object sender, DataObjectPastingEventArgs e)
        {
            //TODO 相关业务处理
            // ...
            
            //取消 粘贴命令
            //e.CancelCommand();
        }

        private static void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            if (sender is TextBox textBox)
            {
                Regex regex = new Regex(GetRegexPattern(textBox));
                //e.Handled = !regex.IsMatch(e.Text);
                e.Handled = !regex.IsMatch(textBox.Text.Insert(textBox.SelectionStart, e.Text));
            }
        }
    }
  • 注意:
    • !regex.IsMatch(e.Text) 是检验输入框内输入的每个字符是否符合正则表达式,校验的是单个的字符
    • !regex.IsMatch(textBox.Text.Insert(textBox.SelectionStart, e.Text)); 校验的是输入当前字符后的整个文本信息,校验的是输入框的Text 属性值
    • 具体需要使用哪种方式进行校验,根据实际业务来定。不过一般对于PreviewTextInput 而言,校验输入的字符即可。

使用附加属性:

        <TextBox x:Name="tb" Height="80" Width="320" VerticalContentAlignment="Center" FontSize="46" 
                 InputMethod.IsInputMethodEnabled="False"
                 local:TextBoxAttachedProperties.RegexPattern="^[0-9]+(\.[0-9]{2})?$" >
        </TextBox>

3. 输入限制小结

  • 以 有两位小数的正实数 为例,提供一个校验思路:
  • 1 【字符的校验】通过 PreviewKeyDown 事件实现对所有按键和字符校验,实现只允许输入数字和小数点
    • 或者直接在PreviewTextInput 事件中对于当前输入字符校验,实现只允许输入数字和小数点
  • 2 【数据格式的校验】通过 TextChanged 事件实现对输入后格式的校验,对于不符合格式的予以提示或矫正处理
    • 为啥使用 TextChanged 对输入后的格式进行校验,是因为TextChanged 可以检测文本的变化,而PreviewTextInput 只能检测文本输入的事件,对于使用Backspace 按键删除文本或逻辑赋值引起的文本变化不会触发该事件
  • 注意:分清楚【字符的校验】和【数据格式的校验】的区别
        private static void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
        {
            if (sender is TextBox textBox)
            {
                //校验输入的字符,数字或小数点可以输入
                Regex regex1 = new Regex("^[0-9.]*$");
                e.Handled = !regex1.IsMatch(e.Text);
            }
        }
        private static void TextBox_TextChanged(object sender, TextChangedEventArgs e)
        {
            if (sender is TextBox textBox)
            {
                //校验输入当前字符后 文本的格式
                //有两位小数的正实数:^[0-9]+(\.[0-9]{2})?$
                Regex regex2 = new Regex("^[0-9]+(\\.[0-9]{2})?$");
                var result = regex2.IsMatch(textBox.Text);
                //这里以Debug.WriteLine 的方式予以提示,实际开发时可以显示到错误模板中
                Debug.WriteLine($"textBox 当前文本信息:{textBox.Text}, 是否符合校验格式:{result} ");
            }
        }

以上代码的运行结果:

  • 1 只可输入数字和小数点
  • 2 校验信息(有两位小数的正实数)如下所示
textBox 当前文本信息:., 是否符合校验格式:False 
textBox 当前文本信息:.1, 是否符合校验格式:False 
textBox 当前文本信息:.12, 是否符合校验格式:False 
textBox 当前文本信息:.1, 是否符合校验格式:False 
textBox 当前文本信息:1, 是否符合校验格式:True 
textBox 当前文本信息:1., 是否符合校验格式:False 
textBox 当前文本信息:1.1, 是否符合校验格式:False 
textBox 当前文本信息:1.12, 是否符合校验格式:True 

更深入的数据验证,请查看 WPF 数据验证详解


结语

回到目录页: WPF 知识汇总
希望以上内容可以帮助到大家,如文中有不对之处,还请批评指正。


参考资料:
C# WPF下限制TextBox只输入数字、小数点、删除等键
WPF TextBox限制只能输入数字的两种方法
关于wpf控件的textbox只允许输入整数,小数,且只能允许输入一位小数点
C# Note12:WPF只允许数字的限制性TextBox
WPF TextBox仅输入数字(含小数)
WPF TextBox 只能输入数字,并限制输入大小
WPF TextBox 允许输入数字及字母的工具类
WPF文本框限制

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/944322.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Docker 安装与配置 Nginx

摘要 1、本文全面介绍了如何在 Docker 环境中安装和配置 Nginx 容器。 2、文中详细解释了如何设置 HTTPS 安全连接及配置 Nginx 以实现前后端分离的代理服务。 2、同时&#xff0c;探讨了通过 IP 和域名两种方式访问 Nginx 服务的具体配置方法 3、此外&#xff0c;文章还涵…

Linux学习2.0——vi编辑器和gcc编译器的使用

vi是一种方便的代码编辑器&#xff0c;Linux系统一般是自带的、 vi和vim的区别&#xff1a;vim是vi的升级版&#xff0c;基础功能两者一致&#xff0c;不过在嵌入式开发板中 只有vi没有vim&#xff0c;但是在Ubuntu上我们可以使用vim编辑器&#xff0c;它的功能更加丰富。 这里…

有一个4*5的矩阵如下,要求编写程序计算总和与平均值,并找出其中值最大的那个元素输出,以及其所在的行号和列号。

一、题目 二、代码解析 max_value 被初始化为矩阵的第一个元素 matrix[0][0]。max_row 和 max_col 分别被初始化为 0&#xff0c;表示最大值所在的行和列。s 被初始化为 0&#xff0c;用于累加矩阵中的所有元素。外层循环 for k in range(len(matrix)) 遍历矩阵的每一行。内层…

html + css 淘宝网实战

之前有小伙伴说&#xff0c;淘宝那么牛逼你会写代码&#xff0c;能帮我做一个一样的淘宝网站吗&#xff0c;好呀&#xff0c;看我接下来如何给你做一个淘宝首页。hahh,开个玩笑。。。学习而已。 在进行html css编写之前 先了解下网页的组成和网页元素的尺寸吧 1.网页的组成 …

网神SecFox FastJson反序列化RCE漏洞复现(附脚本)

0x01 产品描述&#xff1a; ‌网神SecFox是奇安信网神信息技术(北京)股份有限公司推出的一款运维安全管理与审计系统‌&#xff0c;集“身份认证、账户管理、权限控制、运维审计”于一体&#xff0c;提供统一运维身份认证、细粒度的权限控制、丰富的运维审计报告、多维度的预警…

MoH:将多头注意力(Multi-Head Attention)作为头注意力混合(Mixture-of-Head Attention)

摘要 https://arxiv.org/pdf/2410.11842? 在本文中&#xff0c;我们对Transformer模型的核心——多头注意力机制进行了升级&#xff0c;旨在提高效率的同时保持或超越先前的准确度水平。我们表明&#xff0c;多头注意力可以表示为求和形式。鉴于并非所有注意力头都具有同等重…

uniapp——微信小程序读取bin文件,解析文件的数据内容(三)

微信小程序读取bin文件内容 读取用户选择bin文件&#xff0c;并解析数据内容&#xff0c;分包发送给蓝牙设备&#xff1b; 文章目录 微信小程序读取bin文件内容读取文件读取内容返回格式 API文档&#xff1a; getFileSystemManager 关于App端读取bin文件&#xff0c;请查看&…

穷举vs暴搜vs深搜vs回溯vs剪枝系列一>组合

题目&#xff1a; 解析&#xff1a; 代码&#xff1a; private List<List<Integer>> ret;private List<Integer> path;private int n,k;public List<List<Integer>> combine(int _n, int _k) {n _n;k _k;path new ArrayList<>();ret…

鸿蒙开发实战之“使用HiLog和HiSysEvent进行日志与系统事件管理”

HiLog和HiSysEvent作为鸿蒙&#xff08;HarmonyOS&#xff09;系统中进行日志记录和系统事件管理的关键组件&#xff0c;为开发者提供了强大的工具来追踪系统行为、调试应用以及监控设备状态。它们不仅简化了日志管理和事件追踪的流程&#xff0c;还提高了开发效率和系统可维护…

Linux(Centos 7.6)yum源配置

yum是rpm包的管理工具&#xff0c;可以自动安装、升级、删除软件包的功能&#xff0c;可以自动解决软件包之间的依赖关系&#xff0c;使得用户更方便软件包的管理。要使用yum必须要进行配置&#xff0c;个人将其分为三类&#xff0c;本地yum源、局域网yum源、第三方yum源&#…

数据中台从centos升级为国产操作系统后,资源增加字段时,提交报500错误

文章目录 背景一、步骤1.分析阶段2.查看nginx3.修改用户&#xff08;也可以修改所有者权限&#xff09; 背景 故障报错&#xff1a; nginx报错信息&#xff1a; 2024/12/19 15:25:31 [crit, 500299#0: *249 onen0 " /var/lib/nginx/tmp/cient body/0000000001" f…

uniapp 前端解决精度丢失的问题 (后端返回分布式id)

原因&#xff1a; 后端使用分布式id, id为19位数&#xff0c;导致精度丢失 &#xff0c;前端解决方法 这个是通过浏览器请求回来的数据&#xff0c;这个时候id 数据已经丢失了&#xff0c;在数据库查询不到&#xff0c;在调获详情接口的时候会有问题 实际的&#xff1a; 解决…

十大排序---下

文章目录 前言一、归并排序二、快速排序三、计数排序四、桶排序五、基数排序总结 前言 今天我们来继续学习十大排序中剩下的五个。 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、归并排序 归并排序&#xff08;Merge sort&#xff09;是建立在归…

Git如何设置和修改当前分支跟踪的上游分支

目录 前言 背景 设置当前分支跟踪的上游分支 当前分支已有关联&#xff0c;删除其关联&#xff0c;重新设置上游 常用的分支操作 参考资料 前言 仅做学习记录&#xff0c;侵删 背景 在项目开发过程中&#xff0c;从master新建分支时&#xff0c;会出现没有追踪的上游分…

【笔记】Linux中vim编辑器回忆录

&#xff08;一&#xff09;替换 末行模式中 替换整个文本的某个字符为某个东西 全局替换 &#xff1a;%s/旧字符/新字符/g &#xff1a;进入命令行 % 全局范围 s 替换命令 /旧字符/新字符/ 将旧字符换为新字符 g 全局替换 局部范围替换 &#xff1a;开始行号&#xff0c;…

【玩转MacBook】Git安装

Git 官网也提到了MacBook 可以使用 Homebrew 安装 Git&#xff0c;所以在此使用 Homebrew 安装。 1、安装 Homebrew 执行安装脚本 在 Terminal 中执行如下命令&#xff1a; /bin/bash -c "$(curl -fsSL https://gitee.com/ineo6/homebrew-install/raw/master/install.…

Browser Use:AI智能体自动化操作浏览器的开源工具

Browser Use:AI智能体自动化操作浏览器的开源工具 Browser Use 简介1. 安装所需依赖2. 生成openai密钥3. 编写代码4. 运行代码5. 部署与优化5.1 部署AI代理5.2 优化与扩展总结Browser Use 简介 browser-use是一个Python库,它能够帮助我们将AI代理与浏览器自动化操作结合起来;…

字符串存储、分割相关总结(strncpy 函数和strtok() 函数相关)

1.想用这些函数都需要导入头文件 #include<string.h> 2.怎么创建字符串并输入 #define maxsize 100 char a[maxsize1];//创建字符串&#xff0c;预留一个位置放\0 【1】scanf("%s",a);//使用 scanf 函数读取不带空格的字符串 【2】fgets(a, sizeof(a), stdi…

缓存管理自动化:JuiceFS 企业版 Cache Group Operator 新特性发布

近期&#xff0c;JuiceFS 企业版推出了 Cache Group Operator&#xff0c;用于自动化创建和管理缓存组集群。Operator 是一种简化 Kubernetes 应用管理的工具&#xff0c;它能够自动化应用程序的生命周期管理任务&#xff0c;使部署、扩展和运维更加高效。 在推出 Operator 之前…

【蓝桥杯——物联网设计与开发】拓展模块5 - 光敏/热释电模块

目录 一、光敏/热释电模块 &#xff08;1&#xff09;资源介绍 &#x1f505;原理图 &#x1f505;AS312 &#x1f319;简介 &#x1f319;特性 &#x1f505;LDR &#xff08;2&#xff09;STM32CubeMX 软件配置 &#xff08;3&#xff09;代码编写 &#xff08;4&#x…