AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Sprachen und Entwicklungsumgebungen Lazarus (IDE) Numerische Eingabe in Stinggrid validieren
Thema durchsuchen
Ansicht
Themen-Optionen

Numerische Eingabe in Stinggrid validieren

Ein Thema von Oniessen · begonnen am 17. Okt 2014 · letzter Beitrag vom 18. Okt 2014
Antwort Antwort
Oniessen

Registriert seit: 17. Feb 2010
18 Beiträge
 
#1

AW: Numerische Eingabe in Stinggrid validieren

  Alt 17. Okt 2014, 12:58
Hallo nochmal!

Es ging mir ja eigentlich nur um die Besonderheit im TStringGrid. Die andere TEdit-Lösung findet sich recht schnell durch g**geln... und ich dachte eigentlich, das wäre schon recht klar soweit...

Sicher gibts auch Konstanten in Lazarus. Ich denke aber fast, zum Verständniss ist der ASCII-code einfacher...

Naja, hier noch mal die TEdit-Version mit Kommentaren:

Delphi-Quellcode:
procedure TMyForm.MyEditKeyPress(Sender : TObject; var Key : char);
begin
  if not (Key in [#8, '0'..'9', '-', DecimalSeparator]) then Key := #0 // #0 = empty Char/nullCharacter
  // only Keys 0-9(numeric), #8= Backspace, '-' and DecimalSeperator allowed
  else if ((Key = DecimalSeparator) or (Key = '-')) and
          (Pos(Key, (Sender as TEdit).Text) > 0) then Key := #0 // #0 = empty Char/nullCharacter
  // '-' and DecimalSeperator are only allowed once in the String-Input
  else if (Key = '-') and
          ((Sender as TEdit).SelStart <> 0) then Key := #0; // #0 = empty Char/nullCharacter
  // '-' is only allowed as first Character
  // SelStart is the Caret(Text-Curser) Position
end;

Den Typecast zur Veranschaulichung mal anders geschrieben:
(ungetestet)

Delphi-Quellcode:
.
..
selStart:= TStringCellEditor(TStringGrid(Sender).Editor).SelStart ;
// (((Sender as TStringGrid).Editor) as TStringCellEditor).SelStart
..
.

Gruß, Oliver
  Mit Zitat antworten Zitat
Benutzerbild von DeddyH
DeddyH

Registriert seit: 17. Sep 2006
Ort: Barchfeld
27.656 Beiträge
 
Delphi 12 Athens
 
#2

AW: Numerische Eingabe in Stinggrid validieren

  Alt 17. Okt 2014, 13:16
Ich denke aber fast, zum Verständniss ist der ASCII-code einfacher...
Wieso sollte der einfacher sein? Bei entsprechend deklarierten Konstanten schlägt man 2 Fliegen mit einer Klappe: der Code wird besser lesbar und man hat bei evtl. Änderungen nur eine einzige zu ändernde Stelle, nämlich die Konstantendeklaration.
Delphi-Quellcode:
procedure TMyForm.MyEditKeyPress(Sender : TObject; var Key : char);
const
  KEY_NONE = #0;
  KEY_BACKSPACE = #8;
begin
  if not (Key in [KEY_BACKSPACE, '0'..'9', '-', DecimalSeparator]) then Key := KEY_NONE
  else if ((Key = DecimalSeparator) or (Key = '-')) and
          (Pos(Key, (Sender as TEdit).Text) > 0) then Key := KEY_NONE
  else if (Key = '-') and
          ((Sender as TEdit).SelStart <> 0) then Key := KEY_NONE;
  // '-' is only allowed as first Character
  // SelStart is the Caret(Text-Curser) Position
end;
Detlef
"Ich habe Angst vor dem Tag, an dem die Technologie unsere menschlichen Interaktionen übertrumpft. Die Welt wird eine Generation von Idioten bekommen." (Albert Einstein)
Dieser Tag ist längst gekommen
  Mit Zitat antworten Zitat
Blup

Registriert seit: 7. Aug 2008
Ort: Brandenburg
1.487 Beiträge
 
Delphi 12 Athens
 
#3

AW: Numerische Eingabe in Stinggrid validieren

  Alt 17. Okt 2014, 13:46
Wenn ein Teil des Textes aktuell ausgewählt ist, wird er komplett durch das eingegebene Zeichen ersetzt.
Für Vorzeichen und Dezimalseperator muss man den Text also vor der Prüfung kürzen (SelStart, SelLength).
  Mit Zitat antworten Zitat
Oniessen

Registriert seit: 17. Feb 2010
18 Beiträge
 
#4

AW: Numerische Eingabe in Stinggrid validieren

  Alt 17. Okt 2014, 14:00
Hallo Blub!

Das würde ich noch nicht mal als "Fehler" ansehen, denn wenn die Zahl oder ein Teil davon markiert/ausgewählt ist, möchte ich es auch mit was anderem ersetzen. ( meine Meinung)

Da wäre doch eher noch an anderer Stelle abzufangen, das z.b. im StringGrid beim weiter schalten zwischen den Zellen NICHT automatisch markiert wird.

Gruß, Oliver

[EDIT]
OK, habs ausprobiert, wenn nur ein Teil markiert ist, geht's nicht mit dem Vorzeichen

insofern hast du recht, das man das evtl. noch anpassen sollte

[/EDIT]

Geändert von Oniessen (17. Okt 2014 um 14:17 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von DeddyH
DeddyH

Registriert seit: 17. Sep 2006
Ort: Barchfeld
27.656 Beiträge
 
Delphi 12 Athens
 
#5

AW: Numerische Eingabe in Stinggrid validieren

  Alt 17. Okt 2014, 14:00
Das seh ich nicht so, denn die Zeichen werden ja nicht per Code gesetzt, sondern lediglich ggf. entwertet. Wenn der Anwender also einen Teil des Textes markiert, soll dieser ja wohl mit dem eingegebenen Zeichen überschrieben werden. Oder hab ich Dich jetzt falsch verstanden?
Detlef
"Ich habe Angst vor dem Tag, an dem die Technologie unsere menschlichen Interaktionen übertrumpft. Die Welt wird eine Generation von Idioten bekommen." (Albert Einstein)
Dieser Tag ist längst gekommen
  Mit Zitat antworten Zitat
Oniessen

Registriert seit: 17. Feb 2010
18 Beiträge
 
#6

AW: Numerische Eingabe in Stinggrid validieren

  Alt 18. Okt 2014, 08:03
Hallo zusammen!

Ich habe die Lösung noch ein wenig angepaßt, so daß jetzt auch die Eingabe von Vorzeichen bzw. DecimalSeparator bei markiertem Text funktioniert.


Erst das TStringGrid:

Delphi-Quellcode:
// Validates a Float-only input in a Cell of a TStringGrid
procedure TMyForm.MyGridKeyPress(Sender : TObject; var Key : char);
const
  KEY_NONE = #0; // empty Char/nullCharacter
  KEY_BACKSPACE = #8;
var
  CellText:Widestring;
  selStart : integer;
  cellRow, cellCol : integer;
  DecSeparator: char;
begin
   DecSeparator:= DefaultFormatSettings.DecimalSeparator; // the StandAlone Symbol "DecimalSeparator" is deprecated


  // If selection is not empty, first delete this selection, then do the rest
  if TStringCellEditor(TStringGrid(Sender).Editor).SelLength > 0 then
     TStringCellEditor(TStringGrid(Sender).Editor).SelText:= '';


  cellRow:= (sender as TStringGrid).Row;
  cellCol:= (sender as TStringGrid).Col;
  CellText:= (sender as TStringGrid).Cells[cellCol,cellRow];
  selStart:= TStringCellEditor(TStringGrid(Sender).Editor).SelStart ; // exessive Typecasting to access SelStart

  // only Keys 0-9(numeric), #8= Backspace, '-' and DecimalSeperator allowed
  if not (Key in [KEY_BACKSPACE, '0'..'9', '-', DecSeparator])
          then Key := KEY_NONE
  // '-' and DecimalSeperator are only allowed once in the String-Input
  else if ((Key = DecSeparator) or (Key = '-')) and (Pos(Key, CellText) > 0)
          then Key := KEY_NONE
  // '-' is only allowed as first Character / SelStart is the Caret(Text-Curser) Position
  else if (Key = '-') and (selStart <> 0)
          then Key := KEY_NONE
end;

und Hier das Tedit:
Delphi-Quellcode:
// Validates a Float-only input in a TEdit
procedure TMyForm.MyEditKeyPress(Sender : TObject; var Key : char);
const
  KEY_NONE = #0; // empty Char/nullCharacter
  KEY_BACKSPACE = #8;
var
  DecSeparator: Char;
begin
  DecSeparator:= DefaultFormatSettings.DecimalSeparator; // the StandAlone Symbol "DecimalSeparator" is deprecated

  // If selection is not empty, first delete this selection, then do the rest
  if (Sender as TEdit).SelLength > 0 then (Sender as TEdit).SelText:= '';
  

  // only Keys 0-9(numeric), #8= Backspace, '-' and DecimalSeperator allowed
  if not (Key in [KEY_BACKSPACE, '0'..'9', '-', DecSeparator])
          then Key := KEY_NONE
  // '-' and DecimalSeperator are only allowed once in the String-Input
  else if ((Key = DecSeparator) or (Key = '-')) and (Pos(Key, (Sender as TEdit).Text) > 0)
          then Key := KEY_NONE
  // '-' is only allowed as first Character / SelStart is the Caret(Text-Curser) Position
  else if (Key = '-') and ((Sender as TEdit).SelStart <> 0)
          then Key := KEY_NONE
end;
Gruß, Oliver

Geändert von Oniessen (18. Okt 2014 um 08:06 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#7

AW: Numerische Eingabe in Stinggrid validieren

  Alt 18. Okt 2014, 08:09
Per CopyPaste kannst du allerdings noch immer irgendwas in die Zelle bekommen.

Und wenn du in der Zelle -12 stehen hast und vor das - gehst, dann kannst du wieder - drücken und bekommst --12 .

Und diese Eingabe 12,,,,,45 ist auch möglich
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)

Geändert von Sir Rufo (18. Okt 2014 um 08:15 Uhr)
  Mit Zitat antworten Zitat
Benutzerbild von Sir Rufo
Sir Rufo

Registriert seit: 5. Jan 2005
Ort: Stadthagen
9.454 Beiträge
 
Delphi 10 Seattle Enterprise
 
#8

AW: Numerische Eingabe in Stinggrid validieren

  Alt 18. Okt 2014, 09:09
Die einzige zuverlässige Art ist die Verwendung eines Memento und einer Prüfung, ob der aktuelle Wert den Anforderungen entspricht.

Wenn ja, dann wird der aktuelle Wert in das Memento übernommen, wenn nein, dann wird der Wert aus dem Memento wiederhergestellt.
Delphi-Quellcode:
unit EditGuardian;

interface

uses
  System.Generics.Collections,
  System.Classes, System.SysUtils,
  Vcl.StdCtrls;

type
  TEditMemento = class( TPersistent )
  private
    FSelStart: Integer;
    FSelLength: Integer;
    FText: string;
    procedure AssignFromCustomEdit( ASource: TCustomEdit );
    procedure AssignToCustomEdit( ADest: TCustomEdit );
    procedure AssignToEditMemento( ADest: TEditMemento );
  protected
    procedure AssignTo( Dest: TPersistent ); override;
  public
    procedure Assign( Source: TPersistent ); override;
  end;

  TEditGuardian = class
  private
    FEdit: TCustomEdit;
    FMemento: TEditMemento;
    FValidator: TPredicate<string>;
  public
    constructor Create( AEdit: TCustomEdit; AValidator: TPredicate<string> );
    destructor Destroy; override;

    procedure Validate;
  end;

implementation

{ TEditMemento }

procedure TEditMemento.Assign( Source: TPersistent );
begin
  if Source is TCustomEdit
  then
    AssignFromCustomEdit( Source as TCustomEdit )
  else
    inherited;
end;

procedure TEditMemento.AssignFromCustomEdit( ASource: TCustomEdit );
begin
  Self.FText := ASource.Text;
  Self.FSelStart := ASource.SelStart;
  Self.FSelLength := ASource.SelLength;
end;

procedure TEditMemento.AssignTo( Dest: TPersistent );
begin
  if Dest is TCustomEdit
  then
    AssignToCustomEdit( Dest as TCustomEdit )
  else if Dest is TEditMemento
  then
    AssignToEditMemento( Dest as TEditMemento )
  else
    inherited;
end;

procedure TEditMemento.AssignToCustomEdit( ADest: TCustomEdit );
begin
  ADest.Text := Self.FText;
  ADest.SelStart := Self.FSelStart;
  ADest.SelLength := Self.FSelLength;
end;

procedure TEditMemento.AssignToEditMemento( ADest: TEditMemento );
begin
  ADest.FText := Self.FText;
  ADest.FSelStart := Self.FSelStart;
  ADest.FSelLength := Self.FSelLength;
end;

{ TEditGuardian }

constructor TEditGuardian.Create( AEdit: TCustomEdit; AValidator: TPredicate<string> );
begin
  inherited Create;

  if not Assigned( AEdit )
  then
    raise EArgumentNilException.Create( 'AEdit' );
  if not Assigned( AValidator )
  then
    raise EArgumentNilException.Create( 'AValidator' );

  FEdit := AEdit;
  FMemento := TEditMemento.Create;
  FValidator := AValidator;
  Validate;
end;

destructor TEditGuardian.Destroy;
begin
  FMemento.Free;
  inherited;
end;

procedure TEditGuardian.Validate;
begin
  if ( FEdit.Text = '' ) or FValidator( FEdit.Text )
  then
    FMemento.Assign( FEdit )
  else
    FEdit.Assign( FMemento );
end;

end.
Und hier ein kleines Beispiel mit 3 Edits (Integer, Float, Frei)
Delphi-Quellcode:
unit FormMain;

interface

uses

  Winapi.Windows, Winapi.Messages,
  System.SysUtils, System.Variants, System.Classes, System.Generics.Collections,
  Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls, Vcl.AppEvnts,
  EditGuardian;

type
  TMainForm = class( TForm )
    Edit1: TEdit;
    Edit2: TEdit;
    Edit3: TEdit;
    ApplicationEvents1: TApplicationEvents;
    procedure ApplicationEvents1Idle( Sender: TObject; var Done: Boolean );
  private
    FGuardians: TList<TEditGuardian>;
    procedure CheckGuardians;
  public
    procedure AfterConstruction; override;
    procedure BeforeDestruction; override;
  end;

var
  MainForm: TMainForm;

implementation

{$R *.dfm}

procedure TMainForm.AfterConstruction;
begin
  inherited;
  FGuardians := TObjectList<TEditGuardian>.Create;

  // Integer Edit
  FGuardians.Add( TEditGuardian.Create( Edit1,
        function( AValue: string ): Boolean
    var
      LValue: Integer;
    begin
      Result := TryStrToInt( AValue, LValue );
    end ) );

  // FloatEdit
  FGuardians.Add( TEditGuardian.Create( Edit2,
    function( AValue: string ): Boolean
    var
      LValue: Extended;
    begin
      Result := TryStrToFloat( AValue, LValue );
    end ) );

end;

procedure TMainForm.ApplicationEvents1Idle( Sender: TObject; var Done: Boolean );
begin
  CheckGuardians;
end;

procedure TMainForm.BeforeDestruction;
begin
  inherited;
  FGuardians.Free;
end;

procedure TMainForm.CheckGuardians;
var
  LGuardian: TEditGuardian;
begin
  for LGuardian in FGuardians do
    LGuardian.Validate;
end;

end.
Und wie man sieht ist es völlig unerheblich welche Taste auch immer gedrückt wurde, oder auf welchem Weg der Wert es in das Edit-Control geschafft haben könnte. Passt der Wert nicht, wird einfach der alte Zustand wiederhergestellt.

Drops gelutscht

PS Wer jetzt anmerken möchte, dass man dort keine negativen Zahlen eingeben kann, weil ein einfaches - einfach geschluckt wird, der passt entweder den Validator an, oder viel besser erstellt sich eine eigene TryStrToMyValue und berücksichtigt dort diese Eingabe und nimmt diese in den Validator mit auf. Denn irgendwann will man den Wert aus dem Edit wieder auslesen und dann benutzt man einfach diesen TryStrToMyValue .
Kaum macht man's richtig - schon funktioniert's
Zertifikat: Sir Rufo (Fingerprint: ‎ea 0a 4c 14 0d b6 3a a4 c1 c5 b9 dc 90 9d f0 e9 de 13 da 60)

Geändert von Sir Rufo (18. Okt 2014 um 09:25 Uhr)
  Mit Zitat antworten Zitat
Oniessen

Registriert seit: 17. Feb 2010
18 Beiträge
 
#9

AW: Numerische Eingabe in Stinggrid validieren

  Alt 18. Okt 2014, 12:12
Per CopyPaste kannst du allerdings noch immer irgendwas in die Zelle bekommen.

Und wenn du in der Zelle -12 stehen hast und vor das - gehst, dann kannst du wieder - drücken und bekommst --12 .

Und diese Eingabe 12,,,,,45 ist auch möglich
Also bei mir geht das nicht! Ich kann in jede Zelle immer nur genau 1(EIN) '-' und 1(EIN) ',' oder'.' eingeben! Das wird ja über die (Pos(Key, CellText) > 0 ( also wenn schon mehr als 0 da sind) abgefangen...

Mit dem CopyPaste hast du allerdings recht!

Gruß, Oliver
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 05:22 Uhr.
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz