C# TextBox中只允许输入数字的解决方法

本文介绍了一种限制自定义TextBox控件仅接受数字输入的方法,包括处理键盘输入、Ctrl+V粘贴及上下文菜单粘贴操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在定制的TextBox控件中,如果只允许输入数字,需要考虑如下三种情况:
  1. 正常按键输入的字符,包括西文、中文字符等
  2. 通过键盘快捷键方式贴入的文本,即Ctrl+V操作
  3. 通过上下文关联菜单的Mouse操作贴入的文本,即”粘贴“操作
     在探讨的同类文章中,多数只考虑了第1种情况,忽略得了第2、3种常见的操作。本文探讨的处理方法核心思路是重载事件OnKeyPress()和两个法ProcessCmdKey()与WndProc(),并把Ctrl+V、关联菜单的Paste操作统一到键盘录入操作中,从而在OnKeyPress()屏蔽掉非数字键。

1、重载键盘事件OnKeyPress()

     键盘输入的字符可以通过重载TextBox控件的OnKeyPress()事件处理,见如下代码:

     /// <summary>
    /// 屏蔽非数字键
    /// </summary>
    protected override void OnKeyPress(KeyPressEventArgs e)
    {
        base.OnKeyPress(e);

        if (this.ReadOnly)
        {
            return;
        }
        
        // 特殊键, 不处理
        if ((int)e.KeyChar <= 31)
        {
            return;
        }

        // 非数字键, 放弃该输入
        if (!char.IsDigit(e.KeyChar))
        {
            e.Handled = true;
            return;
        }
    }

2、重载命令键处理方法ProcessCmdKey()

      可以在ProcessCmdKey()中捕获快捷键Ctrl+V操作。首先要清除当前的选择文本,然后读取剪切板ClipBoard中的内容,最后通过模拟键盘输入的方式”输入“ClipBoard的内容。需要指出,在ProcessCmdKey()方法中不能使用静态方法SendKeys.Send(),但可以通过控件的WndProc()方法发送字符消息以达到模拟键盘录入的目的。见如下代码:

    /// <summary>
    /// 捕获Ctrl+V快捷键操作
    /// </summary>
    protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
    {
        if (keyData == (Keys)Shortcut.CtrlV)  // 快捷键 Ctrl+V 粘贴操作
        {
            this.ClearSelection();

            string text = Clipboard.GetText();
            for (int k = 0; k < text.Length; k++) // can not use SendKeys.Send
            {
                // 通过消息模拟键盘输入, SendKeys.Send()静态方法不行
                SendCharKey(text[k]);
            }
            return true;
        }
        return base.ProcessCmdKey(ref msg, keyData);

    }
    /// <summary>
    /// 通过消息模拟键盘录入
    /// </summary>
    private void SendCharKey(char c)
    {
        Message msg = new Message();

        msg.HWnd = this.Handle;
        msg.Msg = WM_CHAR;
        msg.WParam = (IntPtr)c;
        msg.LParam = IntPtr.Zero;

        base.WndProc(ref msg);
    }

3、重载消息处理方法WndProc()

     可以在定制TextBox控件中创建无内容的上下文菜单对象,从而屏蔽该菜单,方法是在定制控件的构造函数中增加如下代码:
    public class CustomTextBox: TextBox
    {
        this.ContextMenu = new ConTextMenu();  // 创建无内容菜单对象
    }
     由于上下文菜单的Paste操作对应Windows的WM_PASTE消息,于是可以在控件的WndProc()方法中捕获该消息,然后获得剪切板ClipBoard中的内容,最后通过SendKeys.Send()方法模拟键盘录入操作。需要注意,这里不能调用前面ProcessCmdKey()中模拟键盘输入函数SendCharKey()。见如下代码:

    /// <summary>
    /// 捕获Mouse的Paste消息
    /// </summary>
    protected override void WndProc(ref Message m)
    {
        if (m.Msg == WM_PASTE)  // 选择上下文菜单的"粘贴"操作
        {
            this.ClearSelection();
            SendKeys.Send(Clipboard.GetText());  // 模拟键盘输入
        }
        else
        {
            base.WndProc(ref m);
        }
    }

4、消除选择ClearSelection()、删除字符DeleteText()

     还必须分析前面代码中的函数。其中,函数ClearSelection()用以清除当前的选择文本,即清除this.SelectedText;函数DeleteText()则删除当前字符。注意其中的技巧,就是转换Delete键操作为BackSpace操作。此外,DeleteText()函数还需要确定当前的this.SelectionStart值。具体代码如下:
    /// <summary>
    /// 清除当前TextBox的选择
    /// </summary>
    private void ClearSelection()
    {
        if (this.SelectionLength == 0)
        {
            return;
        }

        int selLength = this.SelectedText.Length;
        this.SelectionStart += this.SelectedText.Length;  // 光标在选择之后
        this.SelectionLength = 0;

        for (int k = 1; k <= selLength; k++)
        {
            this.DeleteText(Keys.Back);
        }
    }

    /// <summary>
    /// 删除当前字符, 并计算SelectionStart值
    /// </summary>
    private void DeleteText(Keys key)
    {
        int selStart = this.SelectionStart;

        if (key == Keys.Delete)  // 转换Delete操作为BackSpace操作
        {
            selStart += 1;
            if (selStart > base.Text.Length)
            {
                return;
            }
        }

        if (selStart == 0 || selStart > base.Text.Length)  // 不需要删除
        {
            return;
        }

        if (selStart == 1 && base.Text.Length == 1)
        {
            base.Text = "";
            base.SelectionStart = 0;
        }
        else  // selStart > 0
        {
            base.Text = base.Text.Substring(0, selStart - 1) +
                base.Text.Substring(selStart, base.Text.Length - selStart);
            base.SelectionStart = selStart - 1;
        }
    }

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值