如何使编辑控件不接受0作为第一个数字?

use*_*565 3 delphi delphi-7

好吧,我有一个只采用数字的编辑(没有字母,没有符号,只有数字).问题是我不希望用户将0作为第一个数字,例如(0239847).

我正在考虑制作一个变量"x" - >字节,它将计算编辑的长度并删除第一个数字,如果它是0:

var l:length; number:string100;
begin
l:length(edit1.text);
Run Code Online (Sandbox Code Playgroud)

现在,如果第一个数字是0(0239847),则删除它,但如果没有其他数字(仅为0),则保持原样.

哦,我终于找到了最简单的方法:

procedure TForm1.Edit1Change(Sender: TObject);
var digit1:string;
begin
digit1:=edit1.Text;
if (digit1='00') or (digit1='01') or (digit1='02') or (digit1='03') or (digit1='04') or
(digit1='05') or (digit1='06') or (digit1='07') or (digit1='08') or (digit1='09') then
edit1.Text:=clear;    //or edit1.text:=0;  it's the same

end;

end.
Run Code Online (Sandbox Code Playgroud)

仍然可以复制和粘贴数字,但我不介意

这是一个非常愚蠢的问题,因为答案很简单明了,但是如果你太累了,很难找到它,我很抱歉让你为此忙,Andreas,感谢你的帮助和提示.

And*_*and 10

很难做到这一点.最雄心勃勃的方法是在按键,粘贴等方面做很多逻辑,就像在这个答案中一样.你会惊讶地发现有多少案件需要处理!

这很好用,我想:

type
  TEdit = class(StdCtrls.TEdit)

...

procedure TEdit.KeyPress(var Key: Char);

  function InvalidKey: boolean;
  begin
     InvalidKey :=
       (
         (Key = '0') and
           (
             ((SelStart = 0) and (Length(Text) - SelLength <> 0))
             or
             ((SelStart = 1) and (Text[1] = '0'))
           )
       )
       or
       (
         not (Key in ['0'..'9', #8, ^Z, ^X, ^C, ^V])
       )
  end;

begin
  inherited;

  if InvalidKey then
  begin
    beep;
    Key := #0;
  end else if (SelStart = 1) and (Text[1] = '0') then
  begin
    Text := Copy(Text, 2);
  end;
end;

procedure TEdit.WMPaste(var Message: TWMPaste);    
var
  ClipbrdText: string;

  function ValidText: boolean;
  var
    i: Integer;
  begin
    result := true;
    for i := 1 to Length(ClipbrdText) do
      if not (ClipbrdText[i] in ['0'..'9']) then
      begin
        result := false;
        break;
      end;
  end;

  function RemoveLeadingZeros(const S: string): string;
  var
    NumLeadingZeros: integer;
    i: Integer;
  begin
    NumLeadingZeros := 0;
    for i := 1 to Length(S) do
      if S[i] = '0' then
        inc(NumLeadingZeros)
      else
        break;
    result := Copy(S, NumLeadingZeros + 1);
  end;

begin
  if Clipboard.HasFormat(CF_TEXT) then
  begin

    ClipbrdText := Clipboard.AsText;

    if not ValidText then
    begin
      beep;
      Exit;
    end;

    if SelStart = 0 then
      ClipbrdText := RemoveLeadingZeros(ClipbrdText);

    SelText := ClipbrdText;

  end
  else
    inherited;
end;
Run Code Online (Sandbox Code Playgroud)

这看起来很复杂,但所有这些逻辑都是必要的.例如,代码可以很好地处理所有这些情况:

  • 您只能输入数字.

  • 如果SelStart = 0不是每一个被选中的性格,你不能进入0,因为这样你会得到一个前导零.

  • 如果SelStart = 0每一个被选中的字符,你可以进入0.因为您应该能够输入此有效数字.

    • 一个特殊情况:如果有零个字符,那么当然选择零个字符,所以通过简单的逻辑你可以0从空字符串的状态进入!
  • 如果SelStart = 1Text[1] = '0'[if SelStart = 1,那么Text[1]存在,所以我们依赖于这里的布尔短路评估],那么你就无法进入0.

  • 如果编辑字段的内容是0并且在零之后添加非零字符,则删除零(并且不会干扰插入符号位置).

  • 您只能粘贴数字.如果SelStart = 0,前导零(在剪贴板数据中)被丢弃.

事实上,为了使行为更加完美,您需要更多代码.例如,假设编辑字段的内容是123000456.就像现在一样,您可以选择前三个字符(123)并按BackspaceDelete获取无效文本000456.解决方案是添加额外的代码,以删除由这两个操作中的任何一个产生的前导零.更新:我一直在思考,并且已经意识到最好不要实现这个(为什么?).

然而,一个捷径就是在没有聚焦的情况下对编辑内容进行"美化",因为当它没有聚焦时,你不太可能通过对内容进行操作来拍摄用户.编辑.

例如,你可以做类似的事情

function PrettifyNumber(const S: string): string;
var
  NumLeadingZeros: integer;
  i: Integer;
begin
  NumLeadingZeros := 0;
  for i := 1 to Length(S) do
    if S[i] = '0' then
      inc(NumLeadingZeros)
    else
      break;
  result := Copy(S, NumLeadingZeros + 1);
  if length(result) = 0 then
    result := S;
end;
Run Code Online (Sandbox Code Playgroud)

然后当它失去键盘焦点时,用其美化版本替换编辑字段的内容.当然,您应该考虑将此功能添加到派生的编辑类中,但是对于测试,您只需使用该OnExit事件:

procedure TForm1.Edit1Exit(Sender: TObject);
begin
  Edit1.Text := PrettifyNumber(Edit1.Text);
end;
Run Code Online (Sandbox Code Playgroud)

我想你想在主要应用程序中进行大量编辑(或者不是那么重要).所以你实际上会编写自己的子类控件.那么我想你还需要其他类型的验证,而不仅仅是删除前导零?因此,您必须将此代码与其余代码很好地集成.如果实际上你将在一个大型应用程序中的任何地方使用这个编辑"子类",你也可以考虑在编辑控件类中添加一个快捷方式,这样你就可以根据需要对文本进行美化/验证,而不仅仅是在退出时(如果编辑控件单独在其表单上,则特别困难!)

更新

根据要求,这是一个非常有效的简单解决方案:

procedure TForm1.Edit1KeyPress(Sender: TObject; var Key: Char);
begin
  if (Key = '0') and (Edit1.SelStart = 0) then Key := #0;
end;
Run Code Online (Sandbox Code Playgroud)

我把它作为练习来解决这种方法的问题.如果你一个接一个地解决问题,当你发现它们时,你最终会得到我的长代码!

  • @ user2296565你怎么没有赞成这个答案?看看安德烈斯在这里为你付出的所有努力.有趣的是,答案获得的投票数通常与所涉及的努力成反比. (4认同)
  • @ user2296565:基于你不客气的态度,你得到的收获超过你应得的.这是一个很棒的帮助.当我需要一些输入验证时,我可能会在将来访问这个答案.感谢Andreas的努力. (4认同)
  • @ user2296565:哪个问题? (2认同)