Componentes.Terceros.jvcl/official/3.39/run/JvTimer.pas
2010-01-18 16:55:50 +00:00

395 lines
10 KiB
ObjectPascal

{-----------------------------------------------------------------------------
The contents of this file are subject to the Mozilla Public License
Version 1.1 (the "License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.mozilla.org/MPL/MPL-1.1.html
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for
the specific language governing rights and limitations under the License.
The Original Code is: JvTimer.PAS, released on 2002-07-04.
The Initial Developers of the Original Code are: Fedor Koshevnikov, Igor Pavluk and Serge Korolev
Copyright (c) 1997, 1998 Fedor Koshevnikov, Igor Pavluk and Serge Korolev
Copyright (c) 2001,2002 SGB Software
All Rights Reserved.
You may retrieve the latest version of this file at the Project JEDI's JVCL home page,
located at http://jvcl.delphi-jedi.org
Known Issues:
-----------------------------------------------------------------------------}
// $Id: JvTimer.pas 12502 2009-09-16 17:35:10Z ahuser $
unit JvTimer;
{$I jvcl.inc}
interface
uses
{$IFDEF UNITVERSIONING}
JclUnitVersioning,
{$ENDIF UNITVERSIONING}
Windows, Messages, SysUtils, ExtCtrls, Classes;
type
TJvTimerEventTime = (tetPre, tetPost);
TJvTimer = class(TComponent)
private
FEnabled: Boolean;
FInterval: Cardinal;
FOnTimer: TNotifyEvent;
FSyncEvent: Boolean;
FThreaded: Boolean;
FTimerThread: TThread;
FTimer: TTimer;
FEventTime: TJvTimerEventTime;
FThreadPriority: TThreadPriority;
FInTimerEvent: Boolean;
procedure SetThreadPriority(Value: TThreadPriority);
procedure SetThreaded(Value: Boolean);
procedure SetEnabled(Value: Boolean);
procedure SetInterval(Value: Cardinal);
procedure SetOnTimer(Value: TNotifyEvent);
procedure UpdateTimer;
protected
procedure DoTimer(Sender: TObject);
procedure Timer; virtual;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
procedure Synchronize(Method: TThreadMethod);
published
property EventTime: TJvTimerEventTime read FEventTime write FEventTime default tetPre;
property Enabled: Boolean read FEnabled write SetEnabled default True;
property Interval: Cardinal read FInterval write SetInterval default 1000;
property SyncEvent: Boolean read FSyncEvent write FSyncEvent default True;
property Threaded: Boolean read FThreaded write SetThreaded default True;
property ThreadPriority: TThreadPriority read FThreadPriority write SetThreadPriority default tpNormal;
property OnTimer: TNotifyEvent read FOnTimer write SetOnTimer;
end;
{$IFDEF UNITVERSIONING}
const
UnitVersioning: TUnitVersionInfo = (
RCSfile: '$URL: https://jvcl.svn.sourceforge.net/svnroot/jvcl/tags/JVCL3_39/run/JvTimer.pas $';
Revision: '$Revision: 12502 $';
Date: '$Date: 2009-09-16 19:35:10 +0200 (mer., 16 sept. 2009) $';
LogPath: 'JVCL\run'
);
{$ENDIF UNITVERSIONING}
implementation
uses
Forms, Consts, SyncObjs,
JvJVCLUtils, JvResources;
//=== { TJvTimerThread } =====================================================
type
TJvTimerThread = class(TThread)
private
FOwner: TJvTimer;
FInterval: Cardinal;
FException: Exception;
FPaused: Boolean;
FPauseSection: TCriticalSection;
FCurrentDuration: Cardinal;
procedure HandleException;
procedure SetPaused(const Value: Boolean);
function GetPaused: Boolean;
protected
procedure Execute; override;
public
constructor Create(Timer: TJvTimer; Enabled: Boolean);
destructor Destroy; override;
property Terminated;
property Paused: Boolean read GetPaused write SetPaused;
end;
constructor TJvTimerThread.Create(Timer: TJvTimer; Enabled: Boolean);
begin
FOwner := Timer;
FPauseSection := TCriticalSection.Create;
inherited Create(not Enabled);
FInterval := 1000;
FreeOnTerminate := False;
end;
procedure TJvTimerThread.HandleException;
begin
if not (FException is EAbort) then
Application.HandleException(Self);
end;
procedure TJvTimerThread.SetPaused(const Value: Boolean);
begin
if FPaused <> Value then
begin
FPauseSection.Acquire;
FPaused := Value;
FPauseSection.Release;
if not FPaused and Suspended then
Suspended := False;
end;
end;
destructor TJvTimerThread.Destroy;
begin
inherited Destroy;
// Used by Execute, and hence in the inherited Destroy (Mantis 3819).
FPauseSection.Free;
end;
procedure TJvTimerThread.Execute;
const
Step = 10; // Time of a wait slot, in milliseconds
var
EventTime: TJvTimerEventTime;
function ThreadClosed: Boolean;
begin
Result := Terminated or Application.Terminated or (FOwner = nil);
end;
begin
repeat
EventTime := FOwner.EventTime;
if EventTime = tetPost then
begin
{ Wait first and then trigger the event }
FCurrentDuration := 0;
while not ThreadClosed and (FCurrentDuration < FInterval) do
begin
SleepEx(Step, False);
Inc(FCurrentDuration, Step);
end;
end;
if not ThreadClosed and not ThreadClosed and FOwner.FEnabled then
begin
if FOwner.SyncEvent then
begin
Synchronize(FOwner.Timer)
end
else
begin
try
FOwner.Timer;
except
on E: Exception do
begin
FException := E;
HandleException;
end;
end;
end;
end;
if EventTime = tetPre then
begin
{ Wait after the event was triggered }
FCurrentDuration := 0;
while not ThreadClosed and (FCurrentDuration < FInterval) do
begin
SleepEx(Step, False);
Inc(FCurrentDuration, Step);
end;
end;
// while we are paused, we do not do anything. However, we do call SleepEx
// in the alertable state to avoid 100% CPU usage. Note that the delay
// should not be 0 as it may lead to 100% CPU in that case. 10 is a safe
// value that is small enough not to have a big impact on restart.
while Paused and not Terminated do
SleepEx(10, True);
until Terminated;
end;
function TJvTimerThread.GetPaused: Boolean;
begin
FPauseSection.Acquire;
Result := FPaused;
FPauseSection.Release;
end;
//=== { TJvTimer } ===========================================================
constructor TJvTimer.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
FEventTime := tetPre;
FEnabled := True;
FInterval := 1000;
FSyncEvent := True;
FThreaded := True;
FThreadPriority := tpNormal;
FTimerThread := nil;
FTimer := nil;
end;
destructor TJvTimer.Destroy;
begin
Destroying;
FEnabled := False;
FOnTimer := nil;
{TTimerThread(FTimerThread).FOwner := nil;}
if Assigned(FTimerThread) then
begin
FTimerThread.Terminate;
(FTimerThread as TJvTimerThread).Paused := False;
FTimerThread.Free;
end;
FTimer.Free;
inherited Destroy;
end;
procedure TJvTimer.DoTimer(Sender: TObject);
begin
Timer;
end;
procedure TJvTimer.UpdateTimer;
begin
if (FInterval <> 0) and FEnabled and Assigned(FOnTimer) then
begin
if FThreaded then
begin
FreeAndNil(FTimer);
if not Assigned(FTimerThread) then
FTimerThread := TJvTimerThread.Create(Self, False);
TJvTimerThread(FTimerThread).Paused := True;
TJvTimerThread(FTimerThread).FCurrentDuration := 0;
TJvTimerThread(FTimerThread).FInterval := FInterval;
FTimerThread.Priority := FThreadPriority;
TJvTimerThread(FTimerThread).Paused := False;
end
else
begin
FreeAndNil(FTimerThread);
if not Assigned(FTimer) then
FTimer := TTimer.Create(Self);
FTimer.Interval := FInterval;
FTimer.OnTimer := DoTimer;
FTimer.Enabled := True;
end;
end
else
begin
{ Don't destroy the thread or the timer if we are currently in the event }
if FInTimerEvent then
begin
if FTimerThread <> nil then
TJvTimerThread(FTimerThread).Paused := True;
if FTimer <> nil then
FTimer.Enabled := False;
Exit;
end;
FreeAndNil(FTimerThread);
FreeAndNil(FTimer);
end;
end;
procedure TJvTimer.SetEnabled(Value: Boolean);
begin
if Value <> FEnabled then
begin
FEnabled := Value;
UpdateTimer;
end;
end;
procedure TJvTimer.SetInterval(Value: Cardinal);
begin
if Value <> FInterval then
begin
FInterval := Value;
UpdateTimer;
end;
end;
procedure TJvTimer.SetThreaded(Value: Boolean);
begin
if Value <> FThreaded then
begin
if FInTimerEvent then
raise Exception.CreateResFmt(@RsCannotChangeInTimerEvent, ['TJvTimer.Threaded']); // do not localize
FThreaded := Value;
UpdateTimer;
end;
end;
procedure TJvTimer.SetThreadPriority(Value: TThreadPriority);
begin
if Value <> FThreadPriority then
begin
FThreadPriority := Value;
if FThreaded then
UpdateTimer;
end;
end;
procedure TJvTimer.Synchronize(Method: TThreadMethod);
begin
if FTimerThread <> nil then
begin
with TJvTimerThread(FTimerThread) do
begin
if Suspended or Terminated then
Method
else
TJvTimerThread(FTimerThread).Synchronize(Method);
end;
end
else
Method;
end;
procedure TJvTimer.SetOnTimer(Value: TNotifyEvent);
begin
if Assigned(FOnTimer) <> Assigned(Value) then
begin
FOnTimer := Value;
UpdateTimer;
end
else
FOnTimer := Value;
end;
procedure TJvTimer.Timer;
begin
if FEnabled and not (csDestroying in ComponentState) and Assigned(FOnTimer) then
begin
FInTimerEvent := True;
try
FOnTimer(Self);
finally
FInTimerEvent := False;
end;
end;
end;
{$IFDEF UNITVERSIONING}
initialization
RegisterUnitVersion(HInstance, UnitVersioning);
finalization
UnregisterUnitVersion(HInstance);
{$ENDIF UNITVERSIONING}
end.