Delphi-PRAXiS
Seite 1 von 2  1 2      

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   GUI-Design mit VCL / FireMonkey / Common Controls (https://www.delphipraxis.net/18-gui-design-mit-vcl-firemonkey-common-controls/)
-   -   Delphi Rekursiven Eventaufruf verhindern (https://www.delphipraxis.net/71262-rekursiven-eventaufruf-verhindern.html)

oXmoX 12. Jun 2006 08:20


Rekursiven Eventaufruf verhindern
 
Hallo Wissende!

Folgendes Problem: Mein Programm fängt das OnClick-Event mehrerer TCheckBoxes ab und verändert bei der Behandlungsroutine wiederum die Checked-Eigenschaft dieser TCheckBoxes. Dadurch wird allerdings das OnClick-Event erneut ausgelöst, was zu einer Endlos-Rekursion und einem Stack-Overflow führt.
Wie kann ich das am geschicktesten verhindern?

Danke und Gruß,
J.

Udontknow 12. Jun 2006 08:44

Re: Rekursiven Eventaufruf verhindern
 
Hallo!

Ob es das geschickteste ist, weiss ich nicht, aber ich schalte vorher einfach eine Variable "Refreshing" auf True, die dem Programm als Merker dient, daß gerade die Steuerelemente vom Programm selbst aktualisiert werden. Diese Variable fragst du im Ereignis dann einfach ab.

Delphi-Quellcode:
Refreshing:=True;
try
  Checkbox1.Checked:=False;
  Checkbox2.Checked:=True;
finally
  Refreshing:=False;
end;
Cu,
Udontknow

michael.lehmann 12. Jun 2006 08:51

Re: Rekursiven Eventaufruf verhindern
 
ganz einfach:


ich gehe mal davon aus das du die checked-eigenschaft im onlick des controls änderst.
in ewta so:
Delphi-Quellcode:
...onclick(sender:TObject)
begin
 for i := 0 to ...
   blablabla
   checked[i] := true;

end;
dabei kommt es natürlich zu onlicks..

lösung:
du musst als erstes den onlick-event stilllegen, dann deinen code durchlaufen und dann wieder aktivieren:

Delphi-Quellcode:
...onclick(sender:TObject)
var
 oce : TNotifyEvent;
begin
 
 oce := (sender as TCheckBox).onlick; // merken
 (sender as TCheckBox).onlick := nil; // stilllegen

 try

  for i := 0 to ...
    blablabla
    checked[i] := true;

 finally
  (sender as TCheckBox).onlick := oce; // wieder aktivierern
 end;

end;
[edit=SirThornberry]Delphi-Tags ergänzt - Mfg, SirThornberry[/edit]

SirThornberry 12. Jun 2006 08:52

Re: Rekursiven Eventaufruf verhindern
 
Delphi-Quellcode:
procedure TYourForm.CheckBoxClicked(Sender: TObject);
var lOnClick: TNotifyEvent;
begin
  lOnClick := TCheckbox(Sender).OnClick;
  TCheckbox(Sender).OnClick := nil;
  //Do whatever you want
  TCheckbox(Sender).OnClick := lOnClick;
end;

Udontknow 12. Jun 2006 08:55

Re: Rekursiven Eventaufruf verhindern
 
@SirThornberry: Das Try/Finally nicht vergessen! :)

oXmoX 12. Jun 2006 09:54

Re: Rekursiven Eventaufruf verhindern
 
oki

danke euch ...so hätt ich es auch gemacht ;)

marabu 12. Jun 2006 11:18

Re: Rekursiven Eventaufruf verhindern
 
Hallo,

hier noch eine andere Vorgehensweise zur Unterbrechung von cascading events:

Delphi-Quellcode:
procedure TDemoForm.CheckBox1Click(Sender: TObject);
begin
  with Sender as TCheckBox do
    if Focused then
    begin
      CheckBox2.Checked := not Checked;
      // ...
    end;
end;
Grüße vom marabu

Udontknow 12. Jun 2006 11:43

Re: Rekursiven Eventaufruf verhindern
 
Ich glaube nicht, daß das Merkmal Focused ausreicht, um sowas zu verhindern. Du kannst doch, weil du gerade diese Checkbox angeklickt hast, eine Aktualisierung dieser Checkbox für nötig befinden...

Cu,
Udontknow

marabu 12. Jun 2006 13:21

Re: Rekursiven Eventaufruf verhindern
 
Hallo,

Zitat:

Zitat von Udontknow
Ich glaube nicht, daß das Merkmal Focused ausreicht, um sowas zu verhindern.

bei sauberem Code tut es das ganz gewiss.

Zitat:

Zitat von Udontknow
Du kannst doch, weil du gerade diese Checkbox angeklickt hast, eine Aktualisierung dieser Checkbox für nötig befinden...

Wenn der User auf eine aktive CheckBox klickt, dann wechselt diese ihren Zustand. Ist das nicht gewollt, so ist die CheckBox zu deaktivieren. Den von dir geschilderten Fall darf es in einem professionell erstellten Programm eigentlich nicht geben. Die von mir vorgestellte Technik verwende ich seit es ereignisgesteuerte frameworks gibt und ich bin ganz sicher nicht der einzige Profi der das so macht.

Freundliche Grüße vom marabu

Lannes 12. Jun 2006 13:37

Re: Rekursiven Eventaufruf verhindern
 
Hallo,

eine Möglichkeit ist hier in der Code-Library beschrieben:
Wert einer Checkbox ändern, ohne OnClick Handler auszulösen

Eine weitere Möglichkeit wird in einem Beitrag in der Entwickler-Ecke/Code-Library aufgezeigt:
Abhängigkeit mehrerer Checkboxen

Die dritte Möglichkeit besteht darin die CheckBox auf 'gechecked' zu setzen ohne die Eigenschaft Checked zu verändern, OnClick wird hierbei nicht ausgelöst:
Delphi-Quellcode:
CheckBox1.Perform(BM_SETCHECK,1,0);
'gechecked' aufheben:
Delphi-Quellcode:
CheckBox1.Perform(BM_SETCHECK,0,0);
Da aber die Eigenschaft Checked nicht verändert wird,
kann folgende Abfrage nicht mehr genutzt werden:
Delphi-Quellcode:
if CheckBox1.Checked then
  //...
Zur Statusabfrage muss dann folgendes genutzt werden:
Delphi-Quellcode:
if CheckBox1.Perform(BM_GETCHECK,0,0) = BST_CHECKED then
  showmessage('checked')
  else
    showmessage('not checked');


Alle Zeitangaben in WEZ +1. Es ist jetzt 22:10 Uhr.
Seite 1 von 2  1 2      

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