2

我正在尝试创建某种许可证验证文本框,该文本框会自动将用户输入拆分为由连字符分隔的块。我的许可证有 25 个字符长,它是这样分开的:

XXXXX-XXXXX-XXXXX-XXXXX-XXXXX

我想出了以下代码来解析用户输入,而他正在键入或通过复制/粘贴通过处理TextChanged文本框控件的事件,如下所示:

public static string InsertStringAtInterval(string source, string insert, int interval)
        {
            StringBuilder result = new StringBuilder();
            int currentPosition = 0;
            while (currentPosition + interval < source.Length)
            {
                result.Append(source.Substring(currentPosition, interval)).Append(insert);
                currentPosition += interval;
            }
            if (currentPosition < source.Length)
            {
                result.Append(source.Substring(currentPosition));
            }
            return result.ToString();
        }

        private bool bHandlingChangeEvent = false;
        private void txtLicense_TextChanged(object sender, EventArgs e)
        {
            if (bHandlingChangeEvent == true)
                return;

            bHandlingChangeEvent = true;
            string text = txtLicense.Text.Replace("-", "").Replace(" ","");
            int nPos = txtLicense.SelectionStart;
            if((text.Length==5||text.Length==10||text.Length==15||text.Length==20) && txtLicense.Text[txtLicense.Text.Length-1]!='-')
            {
                txtLicense.Text += "-";
                txtLicense.SelectionStart = nPos + 1;
            }
            if(text.Length>=25)
            {
                string tmp = text.Substring(0, 25);
                tmp = InsertStringAtInterval(tmp, "-", 5);
                txtLicense.Text = tmp;
                txtLicense.SelectionStart = nPos;
            }


            bHandlingChangeEvent = false;
        }

当我用户在框内键入和粘贴时,这工作得很好。我唯一的问题是,当用户尝试通过按退格键或删除键从输入的键中删除字符时。

由于强制连字符插入@位置 5,10,15,20 一旦用户在退格键上达到这些标记之一,上面的逻辑会强制将连字符添加到字符串中,并且用户不能超出此范围。

我尝试摆弄 KeyDown 事件,但想不出任何有用的东西。有人可以帮忙吗?

我也尝试过使用 MaskedTextbox,但这很丑,因为我不希望掩码/连字符在焦点上可见,而且我当然不想用空格替换提示,因为在单击内部时会产生一些混乱当“假定”为空时,作为光标的框并不总是位于框的开头。

4

3 回答 3

2

这些年来,这种事情一直是我遭受很多痛苦的原因。我处理它的方法是将文本的每个更改都视为从头开始粘贴 - 我去掉连字符,然后将它们放回适当的位置。然后你可以处理用户删除最后一个字符的特殊情况。在这种情况下,您将 TextChanged 事件之前的文本与之后的文本进行比较。如果它们相同,但前面的文本长一个字符,就不要做任何特别的事情。

这里有一些似乎适用于文本输入、剪切/复制/粘贴等的代码。可以通过一些花哨的 LINQ、一些错误处理等来改进它,但希望它能让你的想法得到理解。

private string previousText = string.Empty;
private bool processing = false;

private void textBox1_TextChanged(object sender, EventArgs e)
{
    // If we're already messing around with the text, don't start again.
    if (processing)
        return;

    processing = true;

    // The current textbox text
    string newText = textBox1.Text;

    // What was there before, minus one character.
    string previousLessOne = previousText == string.Empty ? string.Empty : previousText.Substring(0, previousText.Length - 1);

    // Get where the user is, minus any preceding dashes
    int caret = textBox1.SelectionStart - (textBox1.Text.Substring(0, textBox1.SelectionStart).Count(ch => ch == '-'));

    // If the user has done anything other than delete the last character, ensure dashes are placed in the correct position.
    if (newText.Length > 0 && newText != previousLessOne)
    {
        textBox1.Text = string.Empty;
        newText = newText.Replace("-", "");

        for (int i = 0; i < newText.Length; i += 5)
        {
            int length = Math.Min(newText.Length - i, 5);
            textBox1.Text += newText.Substring(i, length);

            if (length == 5)
                textBox1.Text += '-';
        }
    }

    // Put the user back where they were, adjusting for dashes.
    textBox1.SelectionStart = caret + (caret / 5);

    previousText = textBox1.Text;

    processing = false;
}
于 2013-03-06T16:59:12.423 回答
2

你可以试试这个版本:

void txtLicense_KeyDown(object sender, KeyEventArgs e) {
  if (e.KeyCode == Keys.Back) {

    int index = txtLicense.SelectionStart;
    while (index > 0 && txtLicense.Text[index - 1] == '-') {
      --index;
    }

    if (index > 0) {
      --index;
    }

    txtLicense.Select(index, txtLicense.SelectionLength + Math.Max(index, 1));
    txtLicense.SelectedText = string.Empty;

    e.SuppressKeyPress = true;
  }
}
于 2013-03-06T15:47:19.753 回答
1

我以前做过类似的事情。我要做的不是在插入点后面有一个字符之前添加连字符。例如,不要在输入 5 个字符后添加连字符,而是在 6 处添加,这样就不会在末尾添加。

这是我用来在特定位置插入连字符的算法:

public static string FormatLicenseNumber(string str)
{
    if (string.IsNullOrEmpty(str))
        return str;
    else
    {
        str = str.Replace("-", string.Empty);
        if (str.Length > 20)
            str = str.Insert(20, "-");
        if (str.Length > 15)
            str = str.Insert(15, "-");
        if (str.Length > 10)
            str = str.Insert(10, "-");
        if (str.Length > 5)
            str = str.Insert(5, "-");
        return str;
    }
}
于 2013-03-06T15:48:17.390 回答