fpc/rtl/objpas/sysutils/fina.inc
Tomas Hajny 99a9955195 + ExpandFileNameCase implementation added
git-svn-id: trunk@21466 -
2012-06-02 22:30:08 +00:00

436 lines
11 KiB
PHP

{
*********************************************************************
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;