unit uCEFWorkSchedulerThread; {$IFDEF FPC} {$MODE OBJFPC}{$H+} {$ENDIF} {$I cef.inc} {$IFNDEF TARGET_64BITS}{$ALIGN ON}{$ENDIF} {$MINENUMSIZE 4} interface uses {$IFDEF DELPHI16_UP} System.Classes, System.SyncObjs, {$ELSE} Classes, SyncObjs, {$ENDIF} uCEFConstants; type TCEFWorkSchedulerThread = class(TThread) protected FCritSect : TCriticalSection; FInterval : integer; FEvent : TEvent; FOnPulse : TNotifyEvent; FPulsing : boolean; FMustReset : boolean; FDefaultInterval : integer; function Lock : boolean; procedure Unlock; function CanPulse(var aInterval : integer) : boolean; procedure DoOnPulseEvent; procedure EventTimeOut; procedure SignaledEvent; procedure Execute; override; public constructor Create; destructor Destroy; override; procedure AfterConstruction; override; procedure NextPulse(aInterval : integer); property DefaultInterval : integer read FDefaultInterval write FDefaultInterval default CEF_TIMER_MAXDELAY; property OnPulse : TNotifyEvent read FOnPulse write FOnPulse; end; implementation uses uCEFMiscFunctions, {$IFDEF DELPHI16_UP} System.SysUtils, System.Math; {$ELSE} SysUtils, Math; {$ENDIF} constructor TCEFWorkSchedulerThread.Create; begin FOnPulse := nil; FCritSect := nil; FPulsing := False; FEvent := nil; FDefaultInterval := CEF_TIMER_MAXDELAY; FInterval := FDefaultInterval; FMustReset := False; inherited Create(True); FreeOnTerminate := False; end; destructor TCEFWorkSchedulerThread.Destroy; begin if (FEvent <> nil) then FreeAndNil(FEvent); if (FCritSect <> nil) then FreeAndNil(FCritSect); inherited Destroy; end; procedure TCEFWorkSchedulerThread.AfterConstruction; begin inherited AfterConstruction; FEvent := TEvent.Create(nil, False, False, ''); FCritSect := TCriticalSection.Create; end; procedure TCEFWorkSchedulerThread.DoOnPulseEvent; begin if assigned(FOnPulse) then FOnPulse(self); end; function TCEFWorkSchedulerThread.Lock : boolean; begin if not(Terminated) and (FCritSect <> nil) then begin FCritSect.Acquire; Result := True; end else Result := False; end; procedure TCEFWorkSchedulerThread.Unlock; begin if (FCritSect <> nil) then FCritSect.Release; end; procedure TCEFWorkSchedulerThread.NextPulse(aInterval : integer); begin if Lock then try FInterval := min(aInterval, CEF_TIMER_MAXDELAY); FMustReset := True; if FPulsing then begin FPulsing := False; FEvent.SetEvent; end; finally Unlock; end; end; procedure TCEFWorkSchedulerThread.EventTimeOut; begin if Lock then try if FMustReset then begin FInterval := FDefaultInterval; FMustReset := False; end; FPulsing := False; finally Unlock; if not(Terminated) then Synchronize({$IFDEF FPC}self, @{$ENDIF}DoOnPulseEvent); end; end; procedure TCEFWorkSchedulerThread.SignaledEvent; begin if Lock then try FPulsing := False; finally Unlock; end; end; function TCEFWorkSchedulerThread.CanPulse(var aInterval : integer) : boolean; begin Result := False; if Lock then try aInterval := FInterval; if (aInterval > 0) then begin Result := True; FPulsing := True; FEvent.ResetEvent; end; finally Unlock; end; end; procedure TCEFWorkSchedulerThread.Execute; var TempInterval : integer; begin while CanPulse(TempInterval) do if (FEvent.WaitFor(TempInterval) = wrTimeout) then EventTimeOut else SignaledEvent; end; end.