mirror of
https://github.com/doublecmd/doublecmd.git
synced 2026-06-28 10:02:14 +00:00
1204 lines
34 KiB
ObjectPascal
1204 lines
34 KiB
ObjectPascal
{
|
|
Double Commander
|
|
-------------------------------------------------------------------------
|
|
Several useful functions
|
|
|
|
Copyright (C) 2006-2015 Alexander Koblov (alexx2000@mail.ru)
|
|
|
|
contributors:
|
|
|
|
Radek Cervinka <radek.cervinka@centrum.cz>
|
|
(cnvFormatFileSize and DivFileName functions)
|
|
|
|
Tomas Bzatek <tbzatek@users.sourceforge.net>
|
|
(QuoteStr, RemoveQuotation and SplitArgs functions)
|
|
|
|
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
}
|
|
|
|
unit uDCUtils;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, SysUtils, Graphics, Controls, StdCtrls, ColorBox,
|
|
{$IF DEFINED(UNIX)}
|
|
DCBasicTypes,
|
|
{$ENDIF}
|
|
uFile, uTypes;
|
|
|
|
const
|
|
{$IF DEFINED(UNIX)}
|
|
NoQuotesSpecialChars = [' ', '"', '''', '(', ')', ':', '&', '!', '$', '*', '?', '=', '`', '\', #10];
|
|
DoubleQuotesSpecialChars = ['$', '\', '`', '"', #10];
|
|
{$ENDIF}
|
|
|
|
{$IFDEF MSWINDOWS}
|
|
VARDELIMITER='%';
|
|
VARDELIMITER_END='%';
|
|
{$ENDIF}
|
|
|
|
{$IFDEF UNIX }
|
|
VARDELIMITER='$';
|
|
VARDELIMITER_END='';
|
|
{$ENDIF}
|
|
|
|
EnvVarCommanderPath = '%COMMANDER_PATH%'; // Using '%' for backward compatibility
|
|
EnvVarConfigPath = '%DC_CONFIG_PATH%'; // Using '%' for backward compatibility
|
|
EnvVarTodaysDate = VARDELIMITER + 'DC_TODAYSDATE' + VARDELIMITER_END;
|
|
|
|
function GetCmdDirFromEnvVar(const sPath : String) : String;
|
|
function SetCmdDirAsEnvVar(const sPath : String) : String;
|
|
{en
|
|
Replaces environment variables of form %<NAME>% with their values.
|
|
Also replaces the internal "%COMMANDER_PATH%".
|
|
}
|
|
function ReplaceEnvVars(const sText: String): String;
|
|
{en
|
|
Replaces tilde ~ at the beginning of the string with home directory.
|
|
}
|
|
function ReplaceTilde(const Path: String): String;
|
|
{en
|
|
Expands the file name with environment variables by replacing them by absolute path.
|
|
@param(sFileName File name to expand.)
|
|
@returns(Absolute file name.)
|
|
}
|
|
function mbExpandFileName(const sFileName: UTF8String): UTF8String;
|
|
{en
|
|
Convert file size to string representation in floating format (Kb, Mb, Gb)
|
|
@param(iSize File size)
|
|
@param(ShortFormat If @true than short format is used,
|
|
otherwise long format (bytes) is used.)
|
|
@returns(File size in string representation)
|
|
}
|
|
function cnvFormatFileSize(iSize: Int64; FSF: TFileSizeFormat): String;
|
|
function cnvFormatFileSize(iSize: Int64; FSF: Boolean): String;
|
|
function cnvFormatFileSize(iSize: Int64): String; inline;
|
|
{en
|
|
Minimize file path
|
|
@param(PathToMince File path)
|
|
@param(Canvas Output canvas)
|
|
@param(MaxLen Max length of path in pixels)
|
|
@returns(Minimized file path)
|
|
}
|
|
function MinimizeFilePath(const PathToMince: String; Canvas: TCanvas;
|
|
MaxLen: Integer): String;
|
|
{en
|
|
Checks if a filename matches any filename in the filelist or
|
|
if it could be in any directory of the file list or any of their subdirectories.
|
|
@param(Files
|
|
List of files to which the filename must be matched.)
|
|
@param(FileName
|
|
Path to a file that will be matched. This may be absolute, relative
|
|
or contain no path at all (only filename).)
|
|
}
|
|
function MatchesFileList(const Files: TFiles; FileName: String): Boolean;
|
|
{en
|
|
Checks if a file matches any mask in the masklist.
|
|
@param(aFile
|
|
File that will be matched.)
|
|
@param(MaskList
|
|
List of masks to which the file must be matched.)
|
|
}
|
|
function MatchesMaskListEx(const aFile: TFile; MaskList: TStringList): Boolean;
|
|
{en
|
|
Checks if a file or directory belongs in the specified path list.
|
|
Only strings are compared, no file-system checks are done.
|
|
|
|
@param(sBasePathList
|
|
List of absolute paths where the path to check should be in.)
|
|
@param(sPathToCheck
|
|
Absolute path to file or directory to check.)
|
|
@return(@true if sPathToCheck points to a directory or file in sBasePathList.
|
|
@false otherwise.)
|
|
}
|
|
function IsInPathList(sBasePathList : String; sPathToCheck : String;
|
|
ASeparator: AnsiChar = ';') : Boolean;
|
|
{en
|
|
Changes all the files' paths making them relative to 'sNewRootPath'.
|
|
It is done by removing 'sNewRootPath' prefix from the paths and setting
|
|
the general path (Files.Path) to sNewRootPath.
|
|
@param(sNewRootPath
|
|
Path that specifies new 'root' directory for all filenames.)
|
|
@param(Files
|
|
Contains list of files to change.)
|
|
}
|
|
procedure ChangeFileListRoot(sNewRootPath: String; Files: TFiles);
|
|
{en
|
|
Replace executable extension by system specific executable extension
|
|
@param(sFileName File name)
|
|
@returns(Executable name with system specific executable extension)
|
|
}
|
|
function FixExeExt(const sFileName: UTF8String): UTF8String;
|
|
{en
|
|
Delete quotes from string
|
|
@param(Str String)
|
|
}
|
|
function TrimQuotes(const Str: String): String;
|
|
function QuoteStr(const Str: String): String;
|
|
function QuoteFilenameIfNecessary(const Str: String): String;
|
|
function ConcatenateStrWithSpace(const Str: String; const Addition: String):string;
|
|
{$IFDEF UNIX}
|
|
function QuoteSingle(const Str: String): String;
|
|
{$ENDIF}
|
|
function QuoteDouble(const Str: String): String;
|
|
{$IFDEF UNIX}
|
|
{en
|
|
Escapes characters to be inserted between single quotes (')
|
|
and passed to shell command line.
|
|
The resulting string is not enclosed with '', only escaped.
|
|
|
|
For example <cmd1> needs to be escaped with this function:
|
|
sh -c '<cmd1>' "<cmd2>" <cmd3>
|
|
}
|
|
function EscapeSingleQuotes(const Str: String): String;
|
|
{en
|
|
Escapes characters to be inserted between double quotes (")
|
|
and passed to shell command line.
|
|
The resulting string is not enclosed with "", only escaped.
|
|
|
|
For example <cmd2> needs to be escaped with this function:
|
|
sh -c '<cmd1>' "<cmd2>" <cmd3>
|
|
}
|
|
function EscapeDoubleQuotes(const Str: String): String;
|
|
{en
|
|
Escapes characters to be passed to shell command line when no quoting is used.
|
|
|
|
For example <cmd3> needs to be escaped with this function:
|
|
sh -c '<cmd1>' "<cmd2>" <cmd3>
|
|
}
|
|
function EscapeNoQuotes(const Str: String): String;
|
|
{en
|
|
Split command line parameters into argument array
|
|
}
|
|
procedure SplitCommandArgs(const Params: String; out Args: TDynamicStringArray);
|
|
{$ENDIF}
|
|
{en
|
|
Delete quotation characters from string
|
|
@param(Str String)
|
|
@returns(String without quotation characters)
|
|
}
|
|
function RemoveQuotation(const Str: String): String;
|
|
{$IF DEFINED(UNIX)}
|
|
{en
|
|
Split command line to command and a list of arguments.
|
|
Each argument is unquoted.
|
|
@param(sCmdLine Command line)
|
|
@param(sCommand Command)
|
|
@param(Args List of arguments)
|
|
}
|
|
procedure SplitCmdLine(sCmdLine: String; out sCommand: String; out Args: TDynamicStringArray);
|
|
{$ELSEIF DEFINED(MSWINDOWS)}
|
|
{en
|
|
Split command line to command and parameters
|
|
@param(sCmdLine Command line)
|
|
@param(sCmd Command)
|
|
@param(sParams Parameters)
|
|
}
|
|
procedure SplitCmdLine(sCmdLine : String; var sCmd, sParams : String);
|
|
{$ENDIF}
|
|
function CompareStrings(const s1, s2: UTF8String; Natural: Boolean; CaseSensitivity: TCaseSensitivity): PtrInt;
|
|
|
|
procedure InsertFirstItem(sLine: String; comboBox: TCustomComboBox);
|
|
{en
|
|
Compares two strings taking into account the numbers.
|
|
Strings must have tailing zeros (#0).
|
|
}
|
|
function StrFloatCmpW(str1, str2: PWideChar; CaseSensitivity: TCaseSensitivity): PtrInt;
|
|
|
|
function EstimateRemainingTime(StartValue, CurrentValue, EndValue: Int64;
|
|
StartTime: TDateTime; CurrentTime: TDateTime;
|
|
out SpeedPerSecond: Int64): TDateTime;
|
|
|
|
function ModColor(AColor: TColor; APercent: Byte) : TColor;
|
|
{en
|
|
Makes a color some darker
|
|
@param(AColor Source color)
|
|
@param(APercent The percentage of brightness decrease)
|
|
@returns(New some darker color)
|
|
}
|
|
function DarkColor(AColor: TColor; APercent: Byte): TColor;
|
|
{en
|
|
Makes a color some lighter
|
|
@param(AColor Source color)
|
|
@param(APercent The percentage of brightness increase)
|
|
@returns(New some lighter color)
|
|
}
|
|
function LightColor(AColor: TColor; APercent: Byte): TColor;
|
|
procedure SetColorInColorBox(const lcbColorBox: TColorBox; const lColor: TColor);
|
|
procedure UpdateColor(Control: TControl; Checked: Boolean);
|
|
procedure EnableControl(Control: TControl; Enabled: Boolean);
|
|
|
|
procedure SplitCmdLineToCmdParams(sCmdLine : String; var sCmd, sParams : String);
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
LCLProc, LCLType, uMasks, FileUtil, StrUtils, uOSUtils, uGlobs, uGlobsPaths,
|
|
DCStrUtils, DCOSUtils;
|
|
|
|
var
|
|
dtLastDateSubstitutionCheck:TDateTime=0;
|
|
|
|
function GetCmdDirFromEnvVar(const sPath: String): String;
|
|
begin
|
|
Result := NormalizePathDelimiters(sPath);
|
|
Result := StringReplace(Result, EnvVarCommanderPath, ExcludeTrailingPathDelimiter(gpExePath), [rfIgnoreCase]);
|
|
Result := StringReplace(Result, EnvVarConfigPath, ExcludeTrailingPathDelimiter(gpCfgDir), [rfIgnoreCase]);
|
|
end;
|
|
|
|
function SetCmdDirAsEnvVar(const sPath: String): String;
|
|
begin
|
|
Result := NormalizePathDelimiters(sPath);
|
|
Result := StringReplace(Result, ExcludeTrailingPathDelimiter(gpExePath), EnvVarCommanderPath, []);
|
|
Result := StringReplace(Result, ExcludeTrailingPathDelimiter(gpCfgDir), EnvVarConfigPath, []);
|
|
end;
|
|
|
|
function ReplaceEnvVars(const sText: String): String;
|
|
var
|
|
I, X, EqualPos: Integer;
|
|
EnvVar, EnvName, EnvValue: String;
|
|
MyYear, MyMonth, MyDay:word;
|
|
begin
|
|
Result:= sText;
|
|
|
|
//1st, if we have an empty string, get out of here, quick
|
|
if sText = EmptyStr then Exit;
|
|
|
|
//2th, let's check the "easy" substitution, there one related with Double Commander
|
|
if pos('%', sText) > 0 then
|
|
begin
|
|
Result := StringReplace(Result, EnvVarCommanderPath, ExcludeTrailingPathDelimiter(gpExePath), [rfReplaceAll, rfIgnoreCase]);
|
|
Result := StringReplace(Result, EnvVarConfigPath, ExcludeTrailingPathDelimiter(gpCfgDir), [rfReplaceAll, rfIgnoreCase]);
|
|
end;
|
|
|
|
//3nd, if we don't have the variable indication (% in windows for example), get out of here here, quick
|
|
if pos(VARDELIMITER, sText) = 0 then Exit;
|
|
|
|
//4rd, let's check if date changed since last time we updated our dc_todaysdate variable
|
|
if dtLastDateSubstitutionCheck<>Trunc(now) then
|
|
begin
|
|
//Date changed! Let's find where variable is and update it.
|
|
//Don't worry for time consumed: this is done only once per day!
|
|
I:=0;
|
|
while (I<gSpecialDirList.Count) do
|
|
begin
|
|
if gSpecialDirList.SpecialDir[I].VariableName=ENVVARTODAYSDATE then
|
|
begin
|
|
//Variable name found! Let's assign the new date to path value
|
|
DecodeDate(now,MyYear,MyMonth,MyDay);
|
|
gSpecialDirList.SpecialDir[I].PathValue:=Format('%d-%2.2d-%2.2d',[MyYear,MyMonth,MyDay]);
|
|
I:=gSpecialDirList.Count; //To make sure we will end the search loop
|
|
end;
|
|
inc(I);
|
|
end;
|
|
dtLastDateSubstitutionCheck:=Trunc(now); //So we won't re-check this while we're under the same day
|
|
end;
|
|
|
|
//5th, let's roll through the possible variable. We did that with a "while" instead of a constant "for-loop" to get out quickly as soon as we solved the variables
|
|
I:=0;
|
|
while (I<gSpecialDirList.Count) AND (pos(VARDELIMITER,sText)<>0) do
|
|
begin
|
|
if pos(gSpecialDirList.SpecialDir[I].VariableName,Result)<>0 then Result := StringReplace(Result, gSpecialDirList.SpecialDir[I].VariableName, ExcludeTrailingPathDelimiter(gSpecialDirList.SpecialDir[I].PathValue), [rfReplaceAll, rfIgnoreCase]);
|
|
inc(I);
|
|
end;
|
|
|
|
//6th, if we don't have variable indication anymore, (% in windows for example), get out of here here, quick
|
|
if pos(VARDELIMITER, sText)=0 then Exit;
|
|
|
|
//7th, if still we have variable there, let's scan through the environment variable.
|
|
// We got them in the "gSpecialDirList" but just in case some others were added on-the-fly
|
|
// between the moment the application started and the moment we might needed them
|
|
X:= GetEnvironmentVariableCount;
|
|
if X = 0 then Exit; //In the ridiculous possible situation where there is ZERO environment variable...
|
|
I:=1;
|
|
while (I<=X) AND (pos(VARDELIMITER,sText)<>0) do
|
|
begin
|
|
EnvVar:= mbGetEnvironmentString(I);
|
|
EqualPos:= PosEx('=', EnvVar, 2);
|
|
if EqualPos = 0 then Continue;
|
|
EnvName:= Copy(EnvVar, 1, EqualPos - 1);
|
|
EnvValue:= Copy(EnvVar, EqualPos + 1, MaxInt);
|
|
Result:= StringReplace(Result, VARDELIMITER + EnvName + VARDELIMITER_END, EnvValue, [rfReplaceAll, rfIgnoreCase]);
|
|
inc(I);
|
|
end;
|
|
end;
|
|
|
|
function ReplaceTilde(const Path: String): String;
|
|
begin
|
|
{$IFDEF UNIX}
|
|
if StrBegins(Path, '~') and ((Length(Path) = 1) or (Path[2] = PathDelim)) then
|
|
Result := GetHomeDir + Copy(Path, 2, MaxInt)
|
|
else
|
|
{$ENDIF}
|
|
Result := Path;
|
|
end;
|
|
|
|
function mbExpandFileName(const sFileName: UTF8String): UTF8String;
|
|
begin
|
|
Result:= NormalizePathDelimiters(sFileName);
|
|
Result:= ReplaceEnvVars(Result);
|
|
if Pos(PathDelim, Result) <> 0 then
|
|
Result:= ExpandFileName(Result);
|
|
end;
|
|
|
|
function cnvFormatFileSize(iSize: Int64; FSF: TFileSizeFormat): String;
|
|
var
|
|
d: Double;
|
|
begin
|
|
case FSF of
|
|
fsfFloat:
|
|
begin
|
|
if iSize div (1024*1024*1024) > 0 then
|
|
begin
|
|
Result:=FloatToStrF((iSize*16 div (1024*1024*1024))/16, ffFixed, 15, 1)+' G'
|
|
end
|
|
else
|
|
if iSize div (1024*1024) >0 then
|
|
begin
|
|
Result:=FloatToStrF((iSize*10 div (1024*1024))/10, ffFixed, 15, 1)+' M'
|
|
end
|
|
else
|
|
if iSize div 1024 >0 then
|
|
begin
|
|
Result:=FloatToStrF((iSize*10 div 1024)/10, ffFixed, 15, 1)+' K'
|
|
end
|
|
else
|
|
Result:=IntToStr(iSize);
|
|
end;
|
|
fsfB:
|
|
begin
|
|
d:=iSize;
|
|
Result:=Format('%8.0n',[d]);
|
|
end;
|
|
fsfK:
|
|
begin
|
|
Result:=FloatToStrF((iSize*10 div 1024)/10, ffFixed, 15, 1)+' K'
|
|
end;
|
|
fsfM:
|
|
begin
|
|
Result:=FloatToStrF((iSize*10 div (1024*1024))/10, ffFixed, 15, 1)+' M'
|
|
end;
|
|
fsfG:
|
|
begin
|
|
Result:=FloatToStrF((iSize*16 div (1024*1024*1024))/16, ffFixed, 15, 1)+' G'
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function cnvFormatFileSize(iSize: Int64; FSF: Boolean): String;
|
|
begin
|
|
Result := cnvFormatFileSize(iSize, fsfFloat);
|
|
end;
|
|
|
|
function cnvFormatFileSize(iSize: Int64): String;
|
|
begin
|
|
Result := cnvFormatFileSize(iSize, gFileSizeFormat);
|
|
end;
|
|
|
|
{
|
|
This function based on code from http://www.delphirus.com.ru
|
|
}
|
|
|
|
{=========================================================}
|
|
function MinimizeFilePath(const PathToMince: String; Canvas: TCanvas;
|
|
MaxLen: Integer): String;
|
|
{=========================================================}
|
|
// "C:\Program Files\Delphi\DDropTargetDemo\main.pas"
|
|
// "C:\Program Files\..\main.pas"
|
|
Var
|
|
sl: TStringList;
|
|
sHelp, sFile,
|
|
sFirst: String;
|
|
iPos: Integer;
|
|
|
|
Begin
|
|
if MaxLen <= 0 then Exit;
|
|
|
|
sHelp := PathToMince;
|
|
iPos := Pos(PathDelim, sHelp);
|
|
If iPos = 0 Then
|
|
Begin
|
|
Result := PathToMince;
|
|
End
|
|
Else
|
|
Begin
|
|
sl := TStringList.Create;
|
|
// Decode string
|
|
While iPos <> 0 Do
|
|
Begin
|
|
sl.Add(Copy(sHelp, 1, (iPos - 1)));
|
|
sHelp := Copy(sHelp, (iPos + 1), Length(sHelp));
|
|
iPos := Pos(PathDelim, sHelp);
|
|
End;
|
|
If sHelp <> '' Then
|
|
Begin
|
|
sl.Add(sHelp);
|
|
End;
|
|
// Encode string
|
|
sFirst := sl[0];
|
|
sFile := sl[sl.Count - 1];
|
|
sl.Delete(sl.Count - 1);
|
|
Result := '';
|
|
MaxLen := MaxLen - Canvas.TextWidth('XXX');
|
|
if (sl.Count <> 0) and (Canvas.TextWidth(Result + sl[0] + PathDelim + sFile) < MaxLen) then
|
|
begin
|
|
While (sl.Count <> 0) and (Canvas.TextWidth(Result + sl[0] + PathDelim + sFile) < MaxLen) Do
|
|
Begin
|
|
Result := Result + sl[0] + PathDelim;
|
|
sl.Delete(0);
|
|
End;
|
|
If sl.Count = 0 Then
|
|
Begin
|
|
Result := Result + sFile;
|
|
End
|
|
Else
|
|
Begin
|
|
Result := Result + '..' + PathDelim + sFile;
|
|
End;
|
|
end
|
|
else
|
|
If sl.Count = 0 Then
|
|
Begin
|
|
Result := sFirst + PathDelim;
|
|
End
|
|
Else
|
|
Begin
|
|
Result := sFirst + PathDelim + '..' + PathDelim + sFile;
|
|
End;
|
|
sl.Free;
|
|
End;
|
|
//DCDebug('PathX ' , Result);
|
|
if Canvas.TextWidth(Result) > MaxLen + Canvas.TextWidth('XXX') then
|
|
begin
|
|
while (Length(Result) > 0) and (Canvas.TextWidth(Result) > MaxLen) do
|
|
begin
|
|
Delete(Result, Length(Result), 1);
|
|
end;
|
|
Result := Copy(Result, 1, Length(Result) - 3) + '...';
|
|
end;
|
|
End;
|
|
|
|
|
|
function MatchesFileList(const Files: TFiles; FileName: String): Boolean;
|
|
var
|
|
i: Integer;
|
|
aFile: TFile;
|
|
begin
|
|
for i := 0 to Files.Count - 1 do
|
|
begin
|
|
aFile := Files[i];
|
|
|
|
if aFile.IsDirectory then
|
|
begin
|
|
// Check if 'FileName' is in this directory or any of its subdirectories.
|
|
if IsInPath(aFile.FullPath, FileName, True, True) then
|
|
Exit(True);
|
|
end
|
|
else
|
|
begin
|
|
// Item in the list is a file, only compare names.
|
|
if aFile.FullPath = FileName then
|
|
Exit(True);
|
|
end;
|
|
end;
|
|
Result := False;
|
|
end;
|
|
|
|
function MatchesMaskListEx(const aFile: TFile; MaskList: TStringList): Boolean;
|
|
var
|
|
I: Integer;
|
|
sMask,
|
|
sFileName: UTF8String;
|
|
begin
|
|
Result:= False;
|
|
for I:= 0 to MaskList.Count - 1 do
|
|
begin
|
|
sMask:= MaskList[I];
|
|
case GetPathType(sMask) of
|
|
ptAbsolute:
|
|
sFileName:= aFile.FullPath;
|
|
else
|
|
sFileName:= aFile.Name;
|
|
end;
|
|
// When a mask is ended with a PathDelim, it will match only directories
|
|
if (Length(sMask) > 1) and (sMask[Length(sMask)] = PathDelim) then
|
|
begin
|
|
if aFile.IsDirectory then
|
|
sMask:= ExcludeTrailingPathDelimiter(sMask)
|
|
else
|
|
Continue;
|
|
end;
|
|
if MatchesMaskList(sFileName, sMask) then
|
|
Exit(True);
|
|
end;
|
|
end;
|
|
|
|
function IsInPathList(sBasePathList: String; sPathToCheck: String;
|
|
ASeparator: AnsiChar = ';'): Boolean;
|
|
var
|
|
sBasePath: String;
|
|
begin
|
|
sBasePathList := UTF8UpperCase(sBasePathList);
|
|
sPathToCheck := UTF8UpperCase(sPathToCheck);
|
|
repeat
|
|
sBasePath := Copy2SymbDel(sBasePathList, ASeparator);
|
|
if IsInPath(sBasePath, sPathToCheck, True, True) then
|
|
Exit(True);
|
|
until Length(sBasePathList) = 0;
|
|
Result := False
|
|
end;
|
|
|
|
procedure ChangeFileListRoot(sNewRootPath: String; Files: TFiles);
|
|
var
|
|
i: Integer;
|
|
aFile: TFile;
|
|
begin
|
|
if IsInPath(sNewRootPath, Files.Path, True, True) then
|
|
begin
|
|
// Current path is a subpath of new root path.
|
|
|
|
for i := 0 to Files.Count - 1 do
|
|
begin
|
|
aFile := Files[i];
|
|
aFile.Path := ExtractDirLevel(sNewRootPath, aFile.Path);
|
|
end;
|
|
|
|
Files.Path := ExtractDirLevel(sNewRootPath, Files.Path);
|
|
end
|
|
else
|
|
begin
|
|
// Current path has a different base than new root path.
|
|
|
|
if sNewRootPath <> EmptyStr then
|
|
sNewRootPath := IncludeTrailingPathDelimiter(sNewRootPath);
|
|
|
|
for i := 0 to Files.Count - 1 do
|
|
begin
|
|
aFile := Files[i];
|
|
aFile.Path := sNewRootPath + ExtractDirLevel(Files.Path, aFile.Path);
|
|
end;
|
|
|
|
Files.Path := sNewRootPath;
|
|
end;
|
|
end;
|
|
|
|
function FixExeExt(const sFileName: UTF8String): UTF8String;
|
|
var
|
|
ExeExt: UTF8String;
|
|
begin
|
|
Result:= sFileName;
|
|
ExeExt:= GetExeExt;
|
|
if not SameText(ExeExt, ExtractFileExt(sFileName)) then
|
|
Result:= ChangeFileExt(sFileName, ExeExt);
|
|
end;
|
|
|
|
function TrimQuotes(const Str: String): String;
|
|
begin
|
|
Result:= TrimSet(Str, ['"', '''']);
|
|
end;
|
|
|
|
function QuoteStr(const Str: String): String;
|
|
{$IF DEFINED(UNIX)}
|
|
begin
|
|
// Default method is to escape every special char with backslash.
|
|
Result := EscapeNoQuotes(Str);
|
|
end;
|
|
{$ELSE}
|
|
begin
|
|
// On Windows only double quotes can be used for quoting.
|
|
// The double quotes on Windows can be nested, e.g.,
|
|
// "cmd /C "type "Some long file name""" or
|
|
// "cmd /C ""some long file.exe" "long param1" "long param2""
|
|
Result := QuoteDouble(Str);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
function QuoteFilenameIfNecessary(const Str: String): String;
|
|
{$IF DEFINED(UNIX)}
|
|
begin
|
|
// Default method is to escape every special char with backslash.
|
|
Result := EscapeNoQuotes(Str);
|
|
end;
|
|
{$ELSE}
|
|
begin
|
|
if Pos(#32, Str) <> 0 then
|
|
Result := QuoteDouble(Str)
|
|
else
|
|
Result := Str;
|
|
end;
|
|
{$ENDIF}
|
|
|
|
function ConcatenateStrWithSpace(const Str: String; const Addition: String):string;
|
|
begin
|
|
result:=Str;
|
|
if Addition <> EmptyStr then
|
|
if result = EmptyStr then
|
|
result := Addition
|
|
else
|
|
result := result + #32 + Addition;
|
|
end;
|
|
|
|
{$IF DEFINED(UNIX)}
|
|
function QuoteSingle(const Str: String): String;
|
|
begin
|
|
Result := '''' + EscapeSingleQuotes(Str) + '''';
|
|
end;
|
|
{$ENDIF}
|
|
|
|
function QuoteDouble(const Str: String): String;
|
|
begin
|
|
{$IF DEFINED(UNIX)}
|
|
Result := '"' + EscapeDoubleQuotes(Str) + '"';
|
|
{$ELSEIF DEFINED(MSWINDOWS)}
|
|
// Nothing needs to be escaped on Windows, because only double quote (") itself
|
|
// would need to be escaped but there's no standard mechanism for escaping it.
|
|
// It seems every application handles it on their own and CMD doesn't support it at all.
|
|
// Also double quote is a forbidden character on FAT, NTFS.
|
|
Result := '"' + Str + '"';
|
|
{$ENDIF}
|
|
end;
|
|
|
|
{$IF DEFINED(UNIX)}
|
|
function EscapeString(const Str: String; const EscapeChars: TCharSet; const EscapeWith: String): String;
|
|
var
|
|
StartPos: Integer = 1;
|
|
CurPos: Integer = 1;
|
|
begin
|
|
Result := '';
|
|
while CurPos <= Length(Str) do
|
|
begin
|
|
if Str[CurPos] in EscapeChars then
|
|
begin
|
|
Result := Result + Copy(Str, StartPos, CurPos - StartPos) + EscapeWith;
|
|
// The character being quoted will be copied later.
|
|
StartPos := CurPos;
|
|
end;
|
|
Inc(CurPos);
|
|
end;
|
|
Result := Result + Copy(Str, StartPos, CurPos - StartPos);
|
|
end;
|
|
|
|
function EscapeSingleQuotes(const Str: String): String;
|
|
begin
|
|
// Single quotes are strong quotes - no special characters are recognized
|
|
// inside those quotes, so only ' needs to be escaped.
|
|
Result := EscapeString(Str, [''''], '''\''');
|
|
end;
|
|
|
|
function EscapeDoubleQuotes(const Str: String): String;
|
|
begin
|
|
// Double quotes are weak quotes and a few special characters are allowed
|
|
// which need to be escaped.
|
|
Result := EscapeString(Str, DoubleQuotesSpecialChars, '\');
|
|
end;
|
|
|
|
function EscapeNoQuotes(const Str: String): String;
|
|
begin
|
|
// When neither single nor double quotes are used several special characters
|
|
// need to be escaped with backslash (single character quote).
|
|
Result := EscapeString(Str, NoQuotesSpecialChars, '\');
|
|
end;
|
|
|
|
// Helper for RemoveQuotation and SplitCmdLine.
|
|
procedure RemoveQuotationOrSplitCmdLine(sCmdLine: String; out sCommand: String; out Args: TDynamicStringArray; bSplitArgs: Boolean; bNoCmd: Boolean = False);
|
|
var
|
|
I : Integer;
|
|
QuoteChar : Char;
|
|
CurrentArg: String = '';
|
|
DoubleQuotesEscape: Boolean = False;
|
|
|
|
procedure AddArgument;
|
|
begin
|
|
if bSplitArgs then
|
|
begin
|
|
if (bNoCmd = False) and (sCommand = '') then
|
|
sCommand := CurrentArg
|
|
else
|
|
begin
|
|
SetLength(Args, Length(Args) + 1);
|
|
Args[Length(Args) - 1] := CurrentArg;
|
|
end;
|
|
CurrentArg := '';
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
sCommand := '';
|
|
SetLength(Args, 0);
|
|
QuoteChar := #0;
|
|
for I := 1 to Length(sCmdLine) do
|
|
case QuoteChar of
|
|
'\':
|
|
begin
|
|
if sCmdLine[I] <> #10 then
|
|
begin
|
|
if not (sCmdLine[I] in NoQuotesSpecialChars) then
|
|
CurrentArg := CurrentArg + '\';
|
|
CurrentArg := CurrentArg + sCmdLine[I];
|
|
end;
|
|
QuoteChar := #0;
|
|
end;
|
|
'''':
|
|
begin
|
|
if sCmdLine[I] = '''' then
|
|
QuoteChar := #0
|
|
else
|
|
CurrentArg := CurrentArg + sCmdLine[I];
|
|
end;
|
|
'"':
|
|
begin
|
|
if DoubleQuotesEscape then
|
|
begin
|
|
if not (sCmdLine[I] in DoubleQuotesSpecialChars) then
|
|
CurrentArg := CurrentArg + '\';
|
|
CurrentArg := CurrentArg + sCmdLine[I];
|
|
DoubleQuotesEscape := False;
|
|
end
|
|
else
|
|
begin
|
|
case sCmdLine[I] of
|
|
'\':
|
|
DoubleQuotesEscape := True;
|
|
'"':
|
|
QuoteChar := #0;
|
|
else
|
|
CurrentArg := CurrentArg + sCmdLine[I];
|
|
end;
|
|
end;
|
|
end;
|
|
else
|
|
begin
|
|
case sCmdLine[I] of
|
|
'\', '''', '"':
|
|
QuoteChar := sCmdLine[I];
|
|
' ', #9:
|
|
if CurrentArg <> '' then
|
|
AddArgument;
|
|
#10:
|
|
AddArgument;
|
|
else
|
|
CurrentArg := CurrentArg + sCmdLine[I];
|
|
end;
|
|
end;
|
|
end;
|
|
if QuoteChar <> #0 then
|
|
raise EInvalidQuoting.Create;
|
|
if CurrentArg <> '' then
|
|
AddArgument;
|
|
if (not bSplitArgs) then
|
|
sCommand := CurrentArg;
|
|
end;
|
|
|
|
procedure SplitCommandArgs(const Params: String; out Args: TDynamicStringArray);
|
|
var
|
|
Unused: String;
|
|
begin
|
|
RemoveQuotationOrSplitCmdLine(Params, Unused, Args, True, True);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
function RemoveQuotation(const Str: String): String;
|
|
{$IF DEFINED(MSWINDOWS)}
|
|
var
|
|
TrimmedStr: String;
|
|
begin
|
|
if Length(Str) = 0 then
|
|
Result := EmptyStr
|
|
else
|
|
begin
|
|
TrimmedStr := Trim(Str);
|
|
if (TrimmedStr[1] = '"') and (TrimmedStr[Length(TrimmedStr)] = '"') then
|
|
Result := Copy(TrimmedStr, 2, Length(TrimmedStr) - 2)
|
|
else
|
|
Result := Str;
|
|
end;
|
|
end;
|
|
{$ELSEIF DEFINED(UNIX)}
|
|
var
|
|
Args: TDynamicStringArray;
|
|
begin
|
|
RemoveQuotationOrSplitCmdLine(Str, Result, Args, False);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
procedure SplitCmdLineToCmdParams(sCmdLine : String; var sCmd, sParams : String);
|
|
var
|
|
iPos : Integer;
|
|
begin
|
|
if Pos('"', sCmdLine) = 1 then
|
|
begin
|
|
iPos := CharPos('"', sCmdLine, 2);
|
|
if iPos = 0 then
|
|
raise EInvalidQuoting.Create;
|
|
sCmd := Copy(sCmdLine, 2, iPos - 2);
|
|
sParams := Copy(sCmdLine, iPos + 2, Length(sCmdLine) - iPos + 1)
|
|
end
|
|
else
|
|
begin
|
|
iPos := Pos(#32, sCmdLine);
|
|
if iPos <> 0 then
|
|
begin
|
|
sCmd := Copy(sCmdLine, 1, iPos - 1);
|
|
sParams := Copy(sCmdLine, iPos + 1, Length(sCmdLine) - iPos + 1)
|
|
end
|
|
else
|
|
begin
|
|
sCmd := sCmdLine;
|
|
sParams := '';
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
{$IF DEFINED(UNIX)}
|
|
procedure SplitCmdLine(sCmdLine: String; out sCommand: String; out Args: TDynamicStringArray);
|
|
begin
|
|
RemoveQuotationOrSplitCmdLine(sCmdLine, sCommand, Args, True);
|
|
end;
|
|
{$ELSEIF DEFINED(MSWINDOWS)}
|
|
procedure SplitCmdLine(sCmdLine : String; var sCmd, sParams : String);
|
|
begin
|
|
SplitCmdLineToCmdParams(sCmdLine,sCmd,sParams);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
function CompareStrings(const s1, s2: UTF8String; Natural: Boolean; CaseSensitivity: TCaseSensitivity): PtrInt; inline;
|
|
begin
|
|
if Natural then
|
|
Result:= StrFloatCmpW(PWideChar(UTF8Decode(s1)), PWideChar(UTF8Decode(s2)), CaseSensitivity)
|
|
else
|
|
begin
|
|
case CaseSensitivity of
|
|
cstNotSensitive:
|
|
Result := WideCompareText(UTF8Decode(s1), UTF8Decode(s2));
|
|
cstLocale:
|
|
Result := WideCompareStr(UTF8Decode(s1), UTF8Decode(s2));
|
|
cstCharValue:
|
|
Result := SysUtils.CompareStr(S1, S2);
|
|
else
|
|
raise Exception.Create('Invalid CaseSensitivity parameter');
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure InsertFirstItem(sLine: String; comboBox: TCustomComboBox);
|
|
var
|
|
I: Integer = 0;
|
|
begin
|
|
if sLine = EmptyStr then Exit;
|
|
with comboBox.Items do
|
|
begin
|
|
// Use case sensitive search
|
|
while (I < Count) and (CompareStr(Strings[I], sLine) <> 0) do Inc(I);
|
|
|
|
if (I < 0) or (I >= Count) then
|
|
begin
|
|
comboBox.Items.Insert(0, sLine);
|
|
comboBox.ItemIndex := 0;
|
|
end
|
|
else if (I > 0) then
|
|
begin
|
|
comboBox.Items.Move(I, 0);
|
|
// Reset selected item (and combobox text), because Move has destroyed it.
|
|
comboBox.ItemIndex := 0;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function WideStrComp(const Str1, Str2 : WideString): PtrInt;
|
|
var
|
|
counter: SizeInt = 0;
|
|
pstr1, pstr2: PWideChar;
|
|
Begin
|
|
pstr1 := PWideChar(Str1);
|
|
pstr2 := PWideChar(Str2);
|
|
While pstr1[counter] = pstr2[counter] do
|
|
Begin
|
|
if (pstr2[counter] = #0) or (pstr1[counter] = #0) then
|
|
break;
|
|
Inc(counter);
|
|
end;
|
|
Result := ord(pstr1[counter]) - ord(pstr2[counter]);
|
|
end;
|
|
|
|
function StrFloatCmpW(str1, str2: PWideChar; CaseSensitivity: TCaseSensitivity): PtrInt;
|
|
var
|
|
is_digit1, is_digit2: boolean;
|
|
string_result: ptrint = 0;
|
|
number_result: ptrint = 0;
|
|
number1_size: ptrint = 0;
|
|
number2_size: ptrint = 0;
|
|
str_cmp: function(const s1, s2: WideString): PtrInt;
|
|
|
|
function is_digit(c: widechar): boolean; inline;
|
|
begin
|
|
result:= (c in ['0'..'9']);
|
|
end;
|
|
|
|
function is_point(c: widechar): boolean; inline;
|
|
begin
|
|
result:= (c in [',', '.']);
|
|
end;
|
|
|
|
begin
|
|
// Set up compare function
|
|
case CaseSensitivity of
|
|
cstNotSensitive: str_cmp:= WideStringManager.CompareTextWideStringProc;
|
|
cstLocale: str_cmp:= WideStringManager.CompareWideStringProc;
|
|
cstCharValue: str_cmp:= @WideStrComp;
|
|
else
|
|
raise Exception.Create('Invalid CaseSensitivity parameter');
|
|
end;
|
|
|
|
while (true) do
|
|
begin
|
|
// compare string part
|
|
while (true) do
|
|
begin
|
|
if str1^ = #0 then
|
|
begin
|
|
if str2^ <> #0 then
|
|
exit(-1)
|
|
else
|
|
exit(0);
|
|
end;
|
|
|
|
if str2^ = #0 then
|
|
begin
|
|
if str1^ <> #0 then
|
|
exit(+1)
|
|
else
|
|
exit(0);
|
|
end;
|
|
|
|
is_digit1 := is_digit(str1^);
|
|
is_digit2 := is_digit(str2^);
|
|
|
|
if (is_digit1 and is_digit2) then break;
|
|
|
|
if (is_digit1 and not is_digit2) then
|
|
exit(-1);
|
|
|
|
if (is_digit2 and not is_digit1) then
|
|
exit(+1);
|
|
|
|
string_result:= str_cmp(str1^, str2^);
|
|
|
|
if (string_result <> 0) then exit(string_result);
|
|
|
|
inc(str1);
|
|
inc(str2);
|
|
end;
|
|
|
|
// skip leading zeroes for number
|
|
while (str1^ = '0') do
|
|
inc(str1);
|
|
while (str2^ = '0') do
|
|
inc(str2);
|
|
|
|
// compare number before decimal point
|
|
while (true) do
|
|
begin
|
|
is_digit1 := is_digit(str1^);
|
|
is_digit2 := is_digit(str2^);
|
|
|
|
if (not is_digit1 and not is_digit2) then
|
|
break;
|
|
|
|
if ((number_result = 0) and is_digit1 and is_digit2) then
|
|
begin
|
|
if (str1^ > str2^) then
|
|
number_result := +1
|
|
else if (str1^ < str2^) then
|
|
number_result := -1
|
|
else
|
|
number_result := 0;
|
|
end;
|
|
|
|
if (is_digit1) then
|
|
begin
|
|
inc(str1);
|
|
inc(number1_size);
|
|
end;
|
|
|
|
if (is_digit2) then
|
|
begin
|
|
inc(str2);
|
|
inc(number2_size);
|
|
end;
|
|
end;
|
|
|
|
if (number1_size <> number2_size) then
|
|
exit(number1_size - number2_size);
|
|
|
|
if (number_result <> 0) then
|
|
exit(number_result);
|
|
|
|
// if there is a decimal point, compare number after one
|
|
if (is_point(str1^) or is_point(str2^)) then
|
|
begin
|
|
if (is_point(str1^)) then
|
|
inc(str1);
|
|
|
|
if (is_point(str2^)) then
|
|
inc(str2);
|
|
|
|
while (true) do
|
|
begin
|
|
is_digit1 := is_digit(str1^);
|
|
is_digit2 := is_digit(str2^);
|
|
|
|
if (not is_digit1 and not is_digit2) then
|
|
break;
|
|
|
|
if (is_digit1 and not is_digit2) then
|
|
begin
|
|
while (str1^ = '0') do
|
|
inc(str1);
|
|
|
|
if (is_digit(str1^)) then
|
|
exit(+1)
|
|
else
|
|
break;
|
|
end;
|
|
|
|
if (is_digit2 and not is_digit1) then
|
|
begin
|
|
while (str2^ = '0') do
|
|
inc(str2);
|
|
|
|
if (is_digit(str2^)) then
|
|
exit(-1)
|
|
else
|
|
break;
|
|
end;
|
|
|
|
if (str1^ > str2^) then
|
|
exit(+1)
|
|
else if (str1^ < str2^) then
|
|
exit(-1);
|
|
|
|
inc(str1);
|
|
inc(str2);
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function EstimateRemainingTime(StartValue, CurrentValue, EndValue: Int64;
|
|
StartTime: TDateTime; CurrentTime: TDateTime;
|
|
out SpeedPerSecond: Int64): TDateTime;
|
|
var
|
|
Speed: Double;
|
|
begin
|
|
SpeedPerSecond := 0;
|
|
Result := 0;
|
|
|
|
if (CurrentValue > StartValue) and (CurrentTime > StartTime) then
|
|
begin
|
|
Speed := Double(CurrentValue - StartValue) / (CurrentTime - StartTime);
|
|
Result := Double(EndValue - CurrentValue) / Speed;
|
|
SpeedPerSecond := Trunc(Speed) div SecsPerDay;
|
|
end;
|
|
end;
|
|
|
|
function ModColor(AColor: TColor; APercent: Byte) : TColor;
|
|
var
|
|
R, G, B : Byte;
|
|
begin
|
|
RedGreenBlue(ColorToRGB(AColor), R, G, B);
|
|
R := R * APercent div 100;
|
|
G := G * APercent div 100;
|
|
B := B * APercent div 100;
|
|
Result := RGBToColor(R, G, B);
|
|
end;
|
|
|
|
function DarkColor(AColor: TColor; APercent: Byte): TColor;
|
|
var
|
|
R, G, B: Byte;
|
|
begin
|
|
RedGreenBlue(ColorToRGB(AColor), R, G, B);
|
|
R:= R - MulDiv(R, APercent, 100);
|
|
G:= G - MulDiv(G, APercent, 100);
|
|
B:= B - MulDiv(B, APercent, 100);
|
|
Result:= RGBToColor(R, G, B);
|
|
end;
|
|
|
|
function LightColor(AColor: TColor; APercent: Byte): TColor;
|
|
var
|
|
R, G, B: Byte;
|
|
begin
|
|
RedGreenBlue(ColorToRGB(AColor), R, G, B);
|
|
R:= R + MulDiv(255 - R, APercent, 100);
|
|
G:= G + MulDiv(255 - G, APercent, 100);
|
|
B:= B + MulDiv(255 - B, APercent, 100);
|
|
Result:= RGBToColor(R, G, B);
|
|
end;
|
|
|
|
procedure SetColorInColorBox(const lcbColorBox: TColorBox; const lColor: TColor);
|
|
//< select in lcbColorBox lColor if lColor in lcbColorBox else
|
|
// add to lcbColorBox lColor and select him
|
|
var
|
|
I: LongInt;
|
|
begin
|
|
if (lcbColorBox = nil) then Exit; // if lcbColorBox not exist
|
|
|
|
with lcbColorBox do
|
|
begin
|
|
// search lColor in colorbox colorlist
|
|
for I:= 0 to Items.Count - 1 do
|
|
if Colors[I] = lColor then // find color
|
|
begin
|
|
// select color
|
|
Selected:= lColor;
|
|
Exit;
|
|
end;// if for
|
|
|
|
//add items to colorbox list
|
|
Items.Objects[Items.Add('$'+HexStr(lColor, 8))]:= TObject(PtrInt(lColor));
|
|
Selected:= lColor;
|
|
end; // with
|
|
end;
|
|
|
|
procedure UpdateColor(Control: TControl; Checked: Boolean);
|
|
begin
|
|
if Checked then
|
|
Control.Color:= clDefault
|
|
else
|
|
Control.Color:= $FFFFFFFF8000000F;
|
|
end;
|
|
|
|
procedure EnableControl(Control: TControl; Enabled: Boolean);
|
|
begin
|
|
Control.Enabled:= Enabled;
|
|
{$IF DEFINED(LCLWIN32)}
|
|
if Enabled then
|
|
begin
|
|
Control.Color:= clDefault;
|
|
Control.Font.Color:= clDefault;
|
|
end
|
|
else
|
|
begin
|
|
Control.Color:= clBtnFace;
|
|
Control.Font.Color:= clGrayText;
|
|
end;
|
|
{$ENDIF}
|
|
end;
|
|
|
|
end.
|
|
|