{ /*************************************************************************** fileutil.pas ------------ ***************************************************************************/ ***************************************************************************** This file is part of the LazUtils package. See the file COPYING.modifiedLGPL.txt, included in this distribution, for details about the license. ***************************************************************************** } { **************************************************************************** BB: 2013-05-19 Note to developers: This unit should contain functions and procedures to maintain compatibility with Delphi's FileUtil unit. File routines that specifically deal with UTF8 filenames should go into the LazFileUtils unit. ***************************************************************************** } unit FileUtil; {$mode objfpc}{$H+} interface uses Classes, SysUtils, Masks, LazUTF8, LazFileUtils; {$if defined(Windows) or defined(darwin)} {$define CaseInsensitiveFilenames} {$endif} {$IF defined(CaseInsensitiveFilenames) or defined(darwin)} {$DEFINE NotLiteralFilenames} {$ENDIF} const UTF8FileHeader = #$ef#$bb#$bf; FilenamesCaseSensitive = {$IFDEF CaseInsensitiveFilenames}false{$ELSE}true{$ENDIF};// lower and upper letters are treated the same FilenamesLiteral = {$IFDEF NotLiteralFilenames}false{$ELSE}true{$ENDIF};// file names can be compared using = string operator // file attributes and states function CompareFilenames(const Filename1, Filename2: string): integer; inline; function CompareFilenamesIgnoreCase(const Filename1, Filename2: string): integer; inline; function ComparePhysicalFilenames(const Filename1, Filename2: string): integer; function CompareFilenames(Filename1: PChar; Len1: integer; Filename2: PChar; Len2: integer; ResolveLinks: boolean): integer; function FilenameIsAbsolute(const TheFilename: string):boolean; inline; function FilenameIsWinAbsolute(const TheFilename: string):boolean; inline; function FilenameIsUnixAbsolute(const TheFilename: string):boolean; inline; procedure CheckIfFileIsExecutable(const AFilename: string); inline; procedure CheckIfFileIsSymlink(const AFilename: string); inline; function FileIsReadable(const AFilename: string): boolean; inline; function FileIsWritable(const AFilename: string): boolean; inline; function FileIsText(const AFilename: string): boolean; inline; function FileIsText(const AFilename: string; out FileReadable: boolean): boolean; inline; function FileIsExecutable(const AFilename: string): boolean; inline; function FileIsSymlink(const AFilename: string): boolean; inline; function FileIsHardLink(const AFilename: string): boolean; inline; function FileSize(const Filename: string): int64; overload; inline; function GetFileDescription(const AFilename: string): string; inline; function ReadAllLinks(const Filename: string; ExceptionOnError: boolean): string; inline; function TryReadAllLinks(const Filename: string): string; inline; // directories function DirPathExists(const FileName: String): Boolean; inline; function ForceDirectory(DirectoryName: string): boolean; inline; function DeleteDirectory(const DirectoryName: string; OnlyChildren: boolean): boolean; function ProgramDirectory: string; function DirectoryIsWritable(const DirectoryName: string): boolean; inline; // filename parts const PascalFileExt: array[1..3] of string = ('.pas','.pp','.p'); function ExtractFileNameOnly(const AFilename: string): string; inline; function ExtractFileNameWithoutExt(const AFilename: string): string; function CompareFileExt(const Filename, Ext: string; CaseSensitive: boolean): integer; overload; inline; function CompareFileExt(const Filename, Ext: string): integer; overload; inline; function FilenameIsPascalUnit(const Filename: string): boolean; function AppendPathDelim(const Path: string): string; inline; function ChompPathDelim(const Path: string): string; inline; function TrimFilename(const AFilename: string): string; inline; function CleanAndExpandFilename(const Filename: string): string; inline; function CleanAndExpandDirectory(const Filename: string): string; inline; function CreateAbsoluteSearchPath(const SearchPath, BaseDirectory: string): string; function CreateRelativePath(const Filename, BaseDirectory: string; UsePointDirectory: boolean = false; AlwaysRequireSharedBaseFolder: Boolean = True): string; inline; function CreateAbsolutePath(const Filename, BaseDirectory: string): string; function FileIsInPath(const Filename, Path: string): boolean; function FileIsInDirectory(const Filename, Directory: string): boolean; // file search type TSearchFileInPathFlag = ( sffDontSearchInBasePath, sffSearchLoUpCase ); TSearchFileInPathFlags = set of TSearchFileInPathFlag; const AllDirectoryEntriesMask = '*'; function GetAllFilesMask: string; inline; function GetExeExt: string; inline; function SearchFileInPath(const Filename, BasePath, SearchPath, Delimiter: string; Flags: TSearchFileInPathFlags): string; function SearchAllFilesInPath(const Filename, BasePath, SearchPath, Delimiter: string; Flags: TSearchFileInPathFlags): TStrings; function FindDiskFilename(const Filename: string): string; function FindDiskFileCaseInsensitive(const Filename: string): string; function FindDefaultExecutablePath(const Executable: string; const BaseDir: string = ''): string; {$IFDEF darwin} function GetDarwinSystemFilename(Filename: string): string; inline; {$ENDIF} type { TFileIterator } TFileIterator = class private FPath: String; FLevel: Integer; FFileInfo: TSearchRec; FSearching: Boolean; function GetFileName: String; public procedure Stop; function IsDirectory: Boolean; public property FileName: String read GetFileName; property FileInfo: TSearchRec read FFileInfo; property Level: Integer read FLevel; property Path: String read FPath; property Searching: Boolean read FSearching; end; TFileFoundEvent = procedure (FileIterator: TFileIterator) of object; TDirectoryFoundEvent = procedure (FileIterator: TFileIterator) of object; TDirectoryEnterEvent = procedure (FileIterator: TFileIterator) of object; { TFileSearcher } TFileSearcher = class(TFileIterator) private FMaskSeparator: char; FFollowSymLink: Boolean; FOnFileFound: TFileFoundEvent; FOnDirectoryFound: TDirectoryFoundEvent; FOnDirectoryEnter: TDirectoryEnterEvent; FFileAttribute: Word; FDirectoryAttribute: Word; procedure RaiseSearchingError; protected procedure DoDirectoryEnter; virtual; procedure DoDirectoryFound; virtual; procedure DoFileFound; virtual; public constructor Create; procedure Search(ASearchPath: String; ASearchMask: String = ''; ASearchSubDirs: Boolean = True; CaseSensitive: Boolean = False); public property MaskSeparator: char read FMaskSeparator write FMaskSeparator; property FollowSymLink: Boolean read FFollowSymLink write FFollowSymLink; property FileAttribute: Word read FFileAttribute write FFileAttribute default faAnyfile; property DirectoryAttribute: Word read FDirectoryAttribute write FDirectoryAttribute default faDirectory; property OnDirectoryFound: TDirectoryFoundEvent read FOnDirectoryFound write FOnDirectoryFound; property OnFileFound: TFileFoundEvent read FOnFileFound write FOnFileFound; property OnDirectoryEnter: TDirectoryEnterEvent read FOnDirectoryEnter write FOnDirectoryEnter; end; { TListFileSearcher } TListFileSearcher = class(TFileSearcher) private FList: TStrings; protected procedure DoFileFound; override; public constructor Create(AList: TStrings); end; { TListDirectoriesSearcher } TListDirectoriesSearcher = class(TFileSearcher) private FDirectoriesList :TStrings; protected procedure DoDirectoryFound; override; public constructor Create(AList: TStrings); end; function FindAllFiles(const SearchPath: String; SearchMask: String = ''; SearchSubDirs: Boolean = True): TStringList; function FindAllDirectories(const SearchPath: string; SearchSubDirs: Boolean = True): TStringList; // flags for copy type TCopyFileFlag = ( cffOverwriteFile, cffCreateDestDirectory, cffPreserveTime ); TCopyFileFlags = set of TCopyFileFlag; // Copy a file and a whole directory tree function CopyFile(const SrcFilename, DestFilename: string; Flags: TCopyFileFlags=[cffOverwriteFile]): boolean; function CopyFile(const SrcFilename, DestFilename: string; PreserveTime: boolean): boolean; function CopyDirTree(const SourceDir, TargetDir: string; Flags: TCopyFileFlags=[]): Boolean; // file actions function ReadFileToString(const Filename: string): string; function GetTempFilename(const Directory, Prefix: string): string; inline; // basic functions similar to the RTL but working with UTF-8 instead of the // system encoding // AnsiToUTF8 and UTF8ToAnsi need a widestring manager under Linux, BSD, MacOSX // but normally these OS use UTF-8 as system encoding so the widestringmanager // is not needed. function NeedRTLAnsi: boolean; inline;// true if system encoding is not UTF-8 procedure SetNeedRTLAnsi(NewValue: boolean); inline; function UTF8ToSys(const s: string): string; inline;// as UTF8ToAnsi but more independent of widestringmanager function SysToUTF8(const s: string): string; inline;// as AnsiToUTF8 but more independent of widestringmanager function ConsoleToUTF8(const s: string): string; inline;// converts OEM encoded string to UTF8 (used with some Windows specific functions) function UTF8ToConsole(const s: string): string; inline;// converts UTF8 string to console encoding (used by Write, WriteLn) // file operations function FileExistsUTF8(const Filename: string): boolean; inline; function FileAgeUTF8(const FileName: string): Longint; inline; function DirectoryExistsUTF8(const Directory: string): Boolean; inline; function ExpandFileNameUTF8(const FileName: string): string; inline; function ExpandUNCFileNameUTF8(const FileName: string): string; function ExtractShortPathNameUTF8(Const FileName : String) : String; function FindFirstUTF8(const Path: string; Attr: Longint; out Rslt: TSearchRec): Longint; inline; function FindNextUTF8(var Rslt: TSearchRec): Longint; inline; procedure FindCloseUTF8(var F: TSearchrec); inline; function FileSetDateUTF8(const FileName: String; Age: Longint): Longint; inline; function FileGetAttrUTF8(const FileName: String): Longint; inline; function FileSetAttrUTF8(const Filename: String; Attr: longint): Longint; inline; function DeleteFileUTF8(const FileName: String): Boolean; inline; function RenameFileUTF8(const OldName, NewName: String): Boolean; inline; function FileSearchUTF8(const Name, DirList : String; ImplicitCurrentDir : Boolean = True): String; inline; function FileIsReadOnlyUTF8(const FileName: String): Boolean; inline; function GetCurrentDirUTF8: String; inline; function SetCurrentDirUTF8(const NewDir: String): Boolean; inline; function CreateDirUTF8(const NewDir: String): Boolean; inline; function RemoveDirUTF8(const Dir: String): Boolean; inline; function ForceDirectoriesUTF8(const Dir: string): Boolean; inline; function FileOpenUTF8(Const FileName : string; Mode : Integer) : THandle; inline; function FileCreateUTF8(Const FileName : string) : THandle; overload; inline; function FileCreateUTF8(Const FileName : string; Rights: Cardinal) : THandle; overload; inline; // environment function ParamStrUTF8(Param: Integer): string; inline; function GetEnvironmentStringUTF8(Index: Integer): string; inline; function GetEnvironmentVariableUTF8(const EnvVar: string): String; inline; function GetAppConfigDirUTF8(Global: Boolean; Create: boolean = false): string; inline; function GetAppConfigFileUTF8(Global: Boolean; SubDir: boolean = false; CreateDir: boolean = false): string; inline; // other function SysErrorMessageUTF8(ErrorCode: Integer): String; inline; implementation uses {$IFDEF windows} Windows; {$ELSE} Unix; {$ENDIF} {$I fileutil.inc} {$IFDEF windows} {$i winfileutil.inc} {$ELSE} {$i unixfileutil.inc} {$ENDIF} end.