When I add slow code to the OnChange event of TPageControl I run into problems.
If the code is fast and doesn’t take a lot of time, things are fine.
However if the code takes a long time to return +/- 0.5 to 1 second, the PageControl starts to act weird.
If the user changes a page sometimes it doesn’t do anything on the first click, and a second click on the page is required to actually make the change happen.
I’ve kind of sort of fixed this with code like this.
(I’ve simplified it a bit, just to show the idea)
type TDelayProc = procedure(Sender: TObject) of object;
TForm = class(TForm)
...
private
FDelayedSender: TObject;
FDelayedEvent: TDelayProc;
procedure SetDelayedEvent(Value: TDelayProc);
property FDelayedSender: TObject read FDelayedSender write FDelayedSender;
property FDelayedEvent: TDelayProc read FDelayedEvent write SetDelayedEvent;
...
procedure TForm1.SetDelayedEvent(Value: TDelayProc);
begin
Timer1.Active:= false;
FDelayedEvent:= Value;
if Assigned(Value) then Timer1.Active:= true
else DelayedSender:= nil;
end;
procedure TForm1.Timer1Timer(Sender: TObject);
begin
Timer1.Active:= false;
if Assigned(DelayedEvent) then DelayedEvent(DelayedSender);
end;
procedure TForm1.PageControl1Change(Sender: TObject);
begin
if PageControl1.ActivePage = TSPage1 then begin
DelayedSender:= Button1;
DelayedEvent:= Button1Click;
end; {if}
end;
As you can see this is a horrible hack.
The code I’m calling is in QuickReport to prepare a report and MySQL query and such, so I don’t have much control over that.
I’m think there’s some Win32 messaging that I’m messing up by not returning from TPageControl.OnChange fast enough, the delay is definitely shorter than 3 seconds though.
I’ve tried ProcessMessages, but that just made things worse and I don’t want to use a separate thread for this.
How do I fix this so I can use the OnChange event handler like normal
I’m unclear about why you’re using the TTimer stuff. If it were me, I think I’d just
PostMessagea custom message to my form from the OnChange event, so the OnChange handler would return immediately. That would allow the PageControl message flow to behave normally. Then in the Message handler for that custom message I would (1) show/start a progress bar form running on a 2nd thread, (2) start the activity which is taking so much time, and (3) when the time consuming activity finishes, shut down the progress bar.Here’s some code for a threaded progress bar, that I modified from something Peter Below posted years ago. It’s NOT pretty, but users don’t care about that as much as they care about “nothing happening” on the screen.