{ ********************************************************************* Copyright (C) 1997, 1998 Gertjan Schouten See the file COPYING.FPC, included in this distribution, for details about the copyright. 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. ********************************************************************** 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 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;