123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435 |
- {
- *********************************************************************
- Copyright (C) 1997, 1998 Gertjan Schouten
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 2 of the License, or
- (at your option) any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- *********************************************************************
- System Utilities For Free Pascal
- }
- function ChangeFileExt(const FileName, Extension: string): string;
- var
- i : longint;
- EndSep : Set of Char;
- begin
- i := Length(FileName);
- EndSep:=AllowDirectorySeparators+AllowDriveSeparators+[ExtensionSeparator];
- while (I > 0) and not(FileName[I] in EndSep) do
- Dec(I);
- if (I = 0) or (FileName[I] <> ExtensionSeparator) then
- I := Length(FileName)+1;
- Result := Copy(FileName, 1, I - 1) + Extension;
- end;
- function ExtractFilePath(const FileName: string): string;
- var
- i : longint;
- EndSep : Set of Char;
- begin
- i := Length(FileName);
- EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
- while (i > 0) and not (FileName[i] in EndSep) do
- Dec(i);
- If I>0 then
- Result := Copy(FileName, 1, i)
- else
- Result:='';
- end;
- function ExtractFileDir(const FileName: string): string;
- var
- i : longint;
- EndSep : Set of Char;
- begin
- I := Length(FileName);
- EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
- while (I > 0) and not (FileName[I] in EndSep) do
- Dec(I);
- if (I > 1) and (FileName[I] in AllowDirectorySeparators) and
- not (FileName[I - 1] in EndSep) then
- Dec(I);
- Result := Copy(FileName, 1, I);
- end;
- function ExtractFileDrive(const FileName: string): string;
- var
- i,l: longint;
- begin
- Result := '';
- l:=Length(FileName);
- if (l<2) then
- exit;
- If (FileName[2] in AllowDriveSeparators) then
- result:=Copy(FileName,1,2)
- else if (FileName[1] in AllowDirectorySeparators) and
- (FileName[2] in AllowDirectorySeparators) then
- begin
- i := 2;
-
- { skip share }
- While (i<l) and Not (Filename[i+1] in AllowDirectorySeparators) do
- inc(i);
- inc(i);
-
- While (i<l) and Not (Filename[i+1] in AllowDirectorySeparators) do
- inc(i);
- Result:=Copy(FileName,1,i);
- end;
- end;
- function ExtractFileName(const FileName: string): string;
- var
- i : longint;
- EndSep : Set of Char;
- begin
- I := Length(FileName);
- EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
- while (I > 0) and not (FileName[I] in EndSep) do
- Dec(I);
- Result := Copy(FileName, I + 1, MaxInt);
- end;
- function ExtractFileExt(const FileName: string): string;
- var
- i : longint;
- EndSep : Set of Char;
- begin
- I := Length(FileName);
- EndSep:=AllowDirectorySeparators+AllowDriveSeparators+[ExtensionSeparator];
- while (I > 0) and not (FileName[I] in EndSep) do
- Dec(I);
- if (I > 0) and (FileName[I] = ExtensionSeparator) then
- Result := Copy(FileName, I, MaxInt)
- else
- Result := '';
- end;
- function ExtractShortPathName(Const FileName : String) : String;
- begin
- {$ifdef MSWINDOWS}
- SetLength(Result,Max_Path);
- SetLength(Result,GetShortPathName(PChar(FileName), Pchar(Result),Length(Result)));
- {$else}
- Result:=FileName;
- {$endif}
- end;
- type
- PathStr=string;
- {$DEFINE FPC_FEXPAND_SYSUTILS}
- {$I fexpand.inc}
- function ExpandFileName (Const FileName : string): String;
- Var S : String;
- Begin
- S:=FileName;
- DoDirSeparators(S);
- Result:=Fexpand(S);
- end;
- {$ifndef HASEXPANDUNCFILENAME}
- function ExpandUNCFileName (Const FileName : string): String;
- begin
- Result:=ExpandFileName (FileName);
- //!! Here should follow code to replace the drive: part with UNC...
- end;
- {$endif HASEXPANDUNCFILENAME}
- function ExpandFileNameCase (const FileName: string; out MatchFound: TFilenameCaseMatch): string;
- var
- SR: TSearchRec;
- ItemsFound: byte;
- FoundPath: string;
- RestPos: SizeUInt;
- Root: string;
- procedure TryCase (const Base, Rest: string);
- var
- SR: TSearchRec;
- RC: longint;
- NextDirPos: SizeUInt;
- NextPart: string;
- NextRest: string;
- SearchBase: string;
- begin
- NextDirPos := 1;
- while (NextDirPos <= Length (Rest)) and
- not (Rest [NextDirPos] in (AllowDirectorySeparators)) do
- Inc (NextDirPos);
- NextPart := Copy (Rest, 1, Pred (NextDirPos));
- {$IFDEF FPC_FEXPAND_DIRSEP_IS_UPDIR}
- if (Length (Rest) >= NextDirPos) and
- (Rest [NextDirPos] in AllowDirectorySeparators) then
- {$ELSE FPC_FEXPAND_DIRSEP_IS_UPDIR}
- while (Length (Rest) >= NextDirPos) and
- (Rest [NextDirPos] in AllowDirectorySeparators) do
- {$ENDIF FPC_FEXPAND_DIRSEP_IS_UPDIR}
- Inc (NextDirPos);
- NextRest := Copy (Rest, NextDirPos, Length (Rest) - Pred (NextDirPos));
- if (Base = '') or (Base [Length (Base)] in AllowDirectorySeparators) then
- SearchBase := Base
- else
- SearchBase := Base + DirectorySeparator;
- RC := FindFirst (SearchBase + AllFilesMask, faAnyFile, SR);
- while (RC = 0) and (ItemsFound < 2) do
- begin
- if UpCase (NextPart) = UpCase (SR.Name) then
- begin
- if Length (NextPart) = Length (Rest) then
- begin
- Inc (ItemsFound);
- if ItemsFound = 1 then
- FoundPath := SearchBase + SR.Name;
- end
- else if SR.Attr and faDirectory = faDirectory then
- TryCase (SearchBase + SR.Name + DirectorySeparator, NextRest);
- end;
- if ItemsFound < 2 then
- RC := FindNext (SR);
- end;
- FindClose (SR);
- end;
- begin
- Result := ExpandFileName (FileName);
- if FileName = '' then
- MatchFound := mkExactMatch
- else
- if (FindFirst (FileName, faAnyFile, SR) = 0) or
- (* Special check for a root directory or a directory with a trailing slash *)
- (* which are not found using FindFirst. *)
- DirectoryExists (FileName) then
- begin
- MatchFound := mkExactMatch;
- Result := ExtractFilePath (Result) + SR.Name;
- FindClose (SR);
- end
- else
- begin
- (* Better close the search handle here before starting the recursive search *)
- FindClose (SR);
- MatchFound := mkNone;
- if FileNameCaseSensitive then
- begin
- ItemsFound := 0;
- FoundPath := '';
- RestPos := Length (ExtractFileDrive (FileName)) + 1;
- if (Length (FileName) > RestPos) then
- begin
- {$IFDEF FPC_FEXPAND_DIRSEP_IS_UPDIR}
- if (Length (FileName) >= RestPos) and
- (FileName [RestPos] in AllowDirectorySeparators) then
- {$ELSE FPC_FEXPAND_DIRSEP_IS_UPDIR}
- while (Length (FileName) >= RestPos) and
- (FileName [RestPos] in AllowDirectorySeparators) do
- {$ENDIF FPC_FEXPAND_DIRSEP_IS_UPDIR}
- Inc (RestPos);
- Root := Copy (FileName, 1, Pred (RestPos));
- TryCase (Root, Copy (FileName, RestPos, Length (FileName) - Length (Root)));
- if ItemsFound > 0 then
- begin
- Result := ExpandFileName (FoundPath);
- if ItemsFound = 1 then
- MatchFound := mkSingleMatch
- else
- MatchFound := mkAmbiguous;
- end;
- end;
- end;
- end;
- end;
- Const
- MaxDirs = 129;
- function ExtractRelativepath (Const BaseName,DestName : String): String;
- Var Source, Dest : String;
- Sc,Dc,I,J : Longint;
- SD,DD : Array[1..MaxDirs] of PChar;
- Const OneLevelBack = '..'+DirectorySeparator;
- begin
- If Uppercase(ExtractFileDrive(BaseName))<>Uppercase(ExtractFileDrive(DestName)) Then
- begin
- Result:=DestName;
- exit;
- end;
- Source:=ExcludeTrailingPathDelimiter(ExtractFilePath(BaseName));
- Dest:=ExcludeTrailingPathDelimiter(ExtractFilePath(DestName));
- SC:=GetDirs (Source,SD);
- DC:=GetDirs (Dest,DD);
- I:=1;
- While (I<=DC) and (I<=SC) do
- begin
- If StrIcomp(DD[i],SD[i])=0 then
- Inc(i)
- else
- Break;
- end;
- Result:='';
- For J:=I to SC do Result:=Result+OneLevelBack;
- For J:=I to DC do Result:=Result+DD[J]+DirectorySeparator;
- Result:=Result+ExtractFileName(DestNAme);
- end;
- Procedure DoDirSeparators (Var FileName : String);
- VAr I : longint;
- begin
- For I:=1 to Length(FileName) do
- If FileName[I] in AllowDirectorySeparators then
- FileName[i]:=DirectorySeparator;
- end;
- Function SetDirSeparators (Const FileName : string) : String;
- begin
- Result:=FileName;
- DoDirSeparators (Result);
- end;
- {
- DirName is split in a #0 separated list of directory names,
- Dirs is an array of pchars, pointing to these directory names.
- The function returns the number of directories found, or -1
- if none were found.
- }
- Function GetDirs (Var DirName : String; Var Dirs : Array of pchar) : Longint;
- Var I : Longint;
- begin
- I:=1;
- Result:=-1;
- While I<=Length(DirName) do
- begin
- If (DirName[i] in AllowDirectorySeparators) and
- { avoid error in case last char=pathdelim }
- (length(dirname)>i) then
- begin
- DirName[i]:=#0;
- Inc(Result);
- Dirs[Result]:=@DirName[I+1];
- end;
- Inc(I);
- end;
- If Result>-1 then inc(Result);
- end;
- function IncludeTrailingPathDelimiter(Const Path : String) : String;
- Var
- l : Integer;
- begin
- Result:=Path;
- l:=Length(Result);
- If (L=0) or not(Result[l] in AllowDirectorySeparators) then
- Result:=Result+DirectorySeparator;
- end;
- function IncludeTrailingBackslash(Const Path : String) : String;
- begin
- Result:=IncludeTrailingPathDelimiter(Path);
- end;
- function ExcludeTrailingBackslash(Const Path: string): string;
- begin
- Result:=ExcludeTrailingPathDelimiter(Path);
- end;
- function ExcludeTrailingPathDelimiter(Const Path: string): string;
- Var
- L : Integer;
- begin
- L:=Length(Path);
- If (L>0) and (Path[L] in AllowDirectorySeparators) then
- Dec(L);
- Result:=Copy(Path,1,L);
- end;
- function IncludeLeadingPathDelimiter(Const Path : String) : String;
- Var
- l : Integer;
- begin
- Result:=Path;
- l:=Length(Result);
- If (L=0) or not(Result[1] in AllowDirectorySeparators) then
- Result:=DirectorySeparator+Result;
- end;
- function ExcludeLeadingPathDelimiter(Const Path: string): string;
- Var
- L : Integer;
- begin
- Result:=Path;
- L:=Length(Result);
- If (L>0) and (Result[1] in AllowDirectorySeparators) then
- Delete(Result,1,1);
- end;
- function IsPathDelimiter(Const Path: string; Index: Integer): Boolean;
- begin
- Result:=(Index>0) and (Index<=Length(Path)) and (Path[Index] in AllowDirectorySeparators);
- end;
- function ConcatPaths(const Paths: array of String): String;
- var
- I: Integer;
- begin
- if Length(Paths) > 0 then
- begin
- Result := Paths[0];
- for I := 1 to Length(Paths) - 1 do
- Result := IncludeTrailingPathDelimiter(Result) + ExcludeLeadingPathDelimiter(Paths[I]);
- end else
- Result := '';
- end;
- Function GetFileHandle(var f : File):THandle;
- begin
- result:=filerec(f).handle;
- end;
- Function GetFileHandle(var f : Text):THandle;
- begin
- result:=textrec(f).handle;
- end;
|