unit DirWatch;
// -----------------------------------------------------------------------------
// Component Name: TDirectoryWatch .
// Module: DirWatch .
// Description: Implements watching for file changes in a designated .
// directory (or directories). .
// Version: 1.4 .
// Date: 10-MAR-2003 .
// Target: Win32, Delphi 3 - Delphi 7 .
// Author: Angus Johnson, angusj-AT-myrealbox-DOT-com .
// A portion of code has been copied from the Drag & Drop .
// Component Suite which I co-authored with Anders Melander. .
// Copyright: � 2003 Angus Johnson .
// .
// Usage: 1. Add a TDirectoryWatch component to your form. .
// 2. Set its Directory property .
// 3. If you wish to watch its subdirectories too then set .
// the WatchSubDir property to true .
// 4. Assign the OnChange event .
// 5. Set Active to true .
// -----------------------------------------------------------------------------
interface
uses
Windows, Messages, SysUtils, Forms, Classes;
type
TNotifyFilters = set of (nfFilename, nfDirname, nfAttrib,
nfSize, nfLastWrite, nfSecurity);
TWatchThread = class; //forward declaration
TDirectoryWatch = class(TComponent)
private
fWindowHandle: THandle;
fWatchThread: TWatchThread;
fWatchSubDirs: boolean;
fDirectory: string;
fActive: boolean;
fNotifyFilters: TNotifyFilters; //see FindFirstChangeNotification in winAPI
fOnChangeEvent: TNotifyEvent;
procedure SetActive(aActive: boolean);
procedure SetDirectory(aDir: string);
procedure SetWatchSubDirs(aWatchSubDirs: boolean);
procedure SetNotifyFilters(aNotifyFilters: TNotifyFilters);
procedure WndProc(var aMsg: TMessage);
public
constructor Create(aOwner: TComponent); override;
destructor Destroy; override;
published
property Directory: string read fDirectory write SetDirectory;
property NotifyFilters: TNotifyFilters
read fNotifyFilters write SetNotifyFilters;
property WatchSubDirs: boolean read fWatchSubDirs write SetWatchSubDirs;
property Active: boolean read fActive write SetActive;
property OnChange: TNotifyEvent read fOnChangeEvent write fOnChangeEvent;
end;
TWatchThread = class(TThread)
private
fOwnerHdl: Thandle;
fChangeNotify : THandle; //Signals whenever Windows detects a change in .
//the watched directory .
fBreakEvent: THandle; //Signals when either the Directory property .
//changes or when the thread terminates .
fDirectory: string;
fWatchSubDirs: longbool;
fNotifyFilters: dword;
fFinished: boolean;
protected
procedure SetDirectory(const Value: string);
procedure ProcessFilenameChanges;
procedure Execute; override;
public
constructor Create( OwnerHdl: THandle;
const InitialDir: string; WatchSubDirs: boolean; NotifyFilters: dword);
destructor Destroy; override;
procedure Terminate;
property Directory: string write SetDirectory;
end;
procedure Register;
implementation
const
NOTIFYCHANGE_MESSAGE = WM_USER + 1;
resourcestring
sInvalidDir = 'Invalid Directory: ';
//----------------------------------------------------------------------------
// Miscellaneous functions ...
//----------------------------------------------------------------------------
procedure Register;
begin
RegisterComponents('Samples', [TDirectoryWatch]);
end;
//----------------------------------------------------------------------------
function DirectoryExists(const Name: string): Boolean;
var
Code: Integer;
begin
Code := GetFileAttributes(PChar(Name));
Result := (Code <> -1) and (FILE_ATTRIBUTE_DIRECTORY and Code <> 0);
end;
//----------------------------------------------------------------------------
// TDirectoryWatch methods ...
//----------------------------------------------------------------------------
constructor TDirectoryWatch.Create(aOwner: TComponent);
begin
inherited Create(aOwner);
//default Notify values - notify if either a file name or a directory name
//changes or if a file is modified ...
fNotifyFilters := [nfFilename, nfDirname, nfLastWrite];
fDirectory := 'C:\';
//this non-visual control needs to handle messages, so ...
if not (csDesigning in ComponentState) then
fWindowHandle := AllocateHWnd(WndProc);
end;
//----------------------------------------------------------------------------
destructor TDirectoryWatch.Destroy;
begin
Active := false;
if not (csDesigning in ComponentState) then
DeallocateHWnd(fWindowHandle);
inherited Destroy;
end;
//----------------------------------------------------------------------------
procedure TDirectoryWatch.WndProc(var aMsg: TMessage);
begin
with aMsg do
if Msg = NOTIFYCHANGE_MESSAGE then
begin
if assigned(OnChange) then OnChange(self);
end else
Result := DefWindowProc(FWindowHandle, Msg, wParam, lParam);
end;
//------------------------------------------------------------------------------
procedure TDirectoryWatch.SetNotifyFilters(aNotifyFilters: TNotifyFilters);
begin
if aNotifyFilters = fNotifyFilters then exit;
fNotifyFilters := aNotifyFilters;
if assigned(fWatchThread) then
begin
Active := false;
Active := true;
end;
end;
//------------------------------------------------------------------------------
procedure TDirectoryWatch.SetWatchSubDirs(aWatchSubDirs: boolean);
begin
if aWatchSubDirs = fWatchSubDirs then exit;
fWatchSubDirs := aWatchSubDirs;
if assigned(fWatchThread) then
begin
Active := false;
Active := true;
end;
end;
//------------------------------------------------------------------------------
procedure TDirectoryWatch.SetDirectory(aDir: string);
begin
if aDir = '' then
begin
Active := false;
fDirectory := '';
exit;
end;
if (aDir[length(aDir)] <> '\') then aDir := aDir + '\';
if aDir = fDirectory then exit;
if not (csDesigning in ComponentState) and not DirectoryExists(aDir) then
raise Exception.Create( sInvalidDir + aDir);
fDirectory := aDir;
if assigned(fWatchThread) then
fWatchThread.Directory := fDirectory;
end;
//------------------------------------------------------------------------------
procedure TDirectoryWatch.SetActive(aActive: boolean);
var
nf: dword;
begin
if aActive = fActive then exit;
fActive := aActive;
if csDesigning in ComponentState then exit;
if fActive then
begin
if not DirectoryExists(fDirectory) then
begin
fActive := false;
raise Exception.Create(sInvalidDir + fDirectory);
end;
nf := 0;
if nfFilename in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_FILE_NAME;
if nfDirname in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_DIR_NAME;
if nfAttrib in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_ATTRIBUTES;
if nfSize in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_SIZE;
if nfLastWrite in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_LAST_WRITE;
if nfSecurity in fNotifyFilters then
nf := nf or FILE_NOTIFY_CHANGE_SECURITY;
fWatchThread := TWatchThread.Create(
fWindowHandle, fDirectory, fWatchSubDirs, nf);
end else
begin
fWatchThread.Terminate;
fWatchThread := nil;
end;
end;
//----------------------------------------------------------------------------
// TWatchThread methods ...
//----------------------------------------------------------------------------
constructor TWatchThread.Create(OwnerHdl: THandle;
const InitialDir: string; WatchSubDirs: boolean; NotifyFilters: dword);
begin
inherited Create(True);
fOwnerHdl := OwnerHdl;
if WatchSubDirs then
cardinal(fWatchSubDirs) := 1 //workaround a Win9x OS issue
else
fWatchSubDirs := false;
FreeOnTerminate := true;
Priority := tpLowest;
fDirectory := InitialDir;
fNotifyFilters := NotifyFilters;
fBreakEvent := windows.CreateEvent(nil, False, False, nil);
Resume;
end;
//------------------------------------------------------------------------------
destructor TWatchThread.Destroy;
begin
CloseHandle(fBreakEvent);
inherited Destroy;
end;
//------------------------------------------------------------------------------
procedure TWatchThread.SetDirectory(const Value: string);
begin
if (Value = FDirectory) then exit;
FDirectory := Value;
SetEvent(fBreakEvent);
end;
//------------------------------------------------------------------------------
procedure TWatchThread.Terminate;
begin
inherited Terminate;
SetEvent(fBreakEvent);
while not fFinished do sleep(10); //avoids a reported resource leak
//if called while closing the application.
end;
//------------------------------------------------------------------------------
procedure TWatchThread.Execute;
begin
//OUTER LOOP - manages Directory property reassignments
while (not Terminated) do
begin
fChangeNotify := FindFirstChangeNotification(pchar(fDirectory),
fWatchSubDirs, fNotifyFilters);
if (fChangeNotify = INVALID_HANDLE_VALUE) then
//Can't monitor the specified directory so we'll just wait for
//a new Directory assignment or the thread terminating ...
WaitForSingleObject(fBreakEvent, INFINITE)
else
try
//Now do the INNER loop...
ProcessFilenameChanges;
finally
FindCloseChangeNotification(fChangeNotify);
end;
end;
fFinished := true;
end;
//------------------------------------------------------------------------------
procedure TWatchThread.ProcessFilenameChanges;
var
WaitResult : DWORD;
HandleArray : array[0..1] of THandle;
const
TEN_MSECS = 10;
HUNDRED_MSECS = 100;
begin
HandleArray[0] := fBreakEvent;
HandleArray[1] := fChangeNotify;
//INNER LOOP - exits only when fBreakEvent signaled
while (not Terminated) do
begin
//waits for either fChangeNotify or fBreakEvent ...
WaitResult := WaitForMultipleObjects(2, @HandleArray, False, INFINITE);
if (WaitResult = WAIT_OBJECT_0 + 1) then //fChangeNotify
begin
repeat //ie: if a number of files are changing in a block
//just post the one notification message ...
FindNextChangeNotification(fChangeNotify);
until Terminated or
(WaitForSingleObject(fChangeNotify, TEN_MSECS) <> WAIT_OBJECT_0);
if Terminated then break;
//OK, now notify the main thread (before restarting inner loop)...
PostMessage(fOwnerHdl, NOTIFYCHANGE_MESSAGE, 0, 0);
end else //fBreakEvent ...
begin
//If the Directory property is undergoing multiple rapid reassignments
//wait 'til this stops before restarting monitoring of a new directory ...
while (not Terminated) and
(WaitForSingleObject(fBreakEvent, HUNDRED_MSECS) = WAIT_OBJECT_0) do;
break; //EXIT LOOP HERE
end;
end;
end;
//------------------------------------------------------------------------------
//------------------------------------------------------------------------------
end.