| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "Common.h"#include "Exceptions.h"#include "CopyParam.h"#include "HierarchicalStorage.h"#include "TextsCore.h"#include "Interface.h"//---------------------------------------------------------------------------const wchar_t * TransferModeNames[] = { L"binary", L"ascii", L"automatic" };const int TransferModeNamesCount = LENOF(TransferModeNames);//---------------------------------------------------------------------------__fastcall TCopyParamType::TCopyParamType(){  Default();}//---------------------------------------------------------------------------__fastcall TCopyParamType::TCopyParamType(const TCopyParamType & Source){  Assign(&Source);}//---------------------------------------------------------------------------__fastcall TCopyParamType::~TCopyParamType(){}//---------------------------------------------------------------------------void __fastcall TCopyParamType::Default(){  // when changing defaults, make sure GetInfoStr() can handle it  FileNameCase = ncNoChange;  PreserveReadOnly = false;  PreserveTime = true;  PreserveTimeDirs = false;  Rights.Number = TRights::rfDefault;  PreserveRights = false; // Was True until #106  IgnorePermErrors = false;  AsciiFileMask.Masks = L"*.*html; *.htm; *.txt; *.php; *.php3; *.cgi; *.c; *.cpp; *.h; *.pas; "    "*.bas; *.tex; *.pl; *.js; .htaccess; *.xtml; *.css; *.cfg; *.ini; *.sh; *.xml";  TransferMode = tmBinary;  AddXToDirectories = true;  ResumeSupport = rsSmart;  ResumeThreshold = 100 * 1024; // (100 KB)  InvalidCharsReplacement = TokenReplacement;  LocalInvalidChars = ::LocalInvalidChars;  CalculateSize = true;  FileMask = L"*.*";  IncludeFileMask.Masks = L"";  TransferSkipList = NULL;  TransferResumeFile = L"";  ClearArchive = false;  RemoveCtrlZ = false;  RemoveBOM = false;  CPSLimit = 0;  NewerOnly = false;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::GetInfoStr(  UnicodeString Separator, int Attrs) const{  UnicodeString Result;  bool SomeAttrIncluded;  UnicodeString ScriptArgs;  bool NoScriptArgs;  UnicodeString AssemblyCode;  bool NoCodeProperties;  DoGetInfoStr(    Separator, Attrs, Result, SomeAttrIncluded,    UnicodeString(), ScriptArgs, NoScriptArgs, TAssemblyLanguage(0), AssemblyCode, NoCodeProperties);  return Result;}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::AnyUsableCopyParam(int Attrs) const{  UnicodeString Result;  bool SomeAttrIncluded;  UnicodeString ScriptArgs;  bool NoScriptArgs;  UnicodeString AssemblyCode;  bool NoCodeProperties;  DoGetInfoStr(    L";", Attrs, Result, SomeAttrIncluded,    UnicodeString(), ScriptArgs, NoScriptArgs, TAssemblyLanguage(0), AssemblyCode, NoCodeProperties);  return SomeAttrIncluded;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::GenerateTransferCommandArgs(int Attrs, const UnicodeString & Link, bool & NoScriptArgs) const{  UnicodeString Result;  bool SomeAttrIncluded;  UnicodeString ScriptArgs;  UnicodeString AssemblyCode;  bool NoCodeProperties;  DoGetInfoStr(    L";", Attrs, Result, SomeAttrIncluded,    Link, ScriptArgs, NoScriptArgs, TAssemblyLanguage(0), AssemblyCode, NoCodeProperties);  return ScriptArgs;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::GenerateAssemblyCode(  TAssemblyLanguage Language, int Attrs, bool & NoCodeProperties) const{  UnicodeString Result;  bool SomeAttrIncluded;  UnicodeString ScriptArgs;  bool NoScriptArgs;  UnicodeString AssemblyCode;  DoGetInfoStr(L";", Attrs, Result, SomeAttrIncluded, UnicodeString(), ScriptArgs, NoScriptArgs, Language, AssemblyCode, NoCodeProperties);  return AssemblyCode;}//---------------------------------------------------------------------------void __fastcall TCopyParamType::DoGetInfoStr(  UnicodeString Separator, int Options,  UnicodeString & Result, bool & SomeAttrIncluded,  const UnicodeString & Link, UnicodeString & ScriptArgs, bool & NoScriptArgs, TAssemblyLanguage Language, UnicodeString & AssemblyCode,  bool & NoCodeProperties) const{  TCopyParamType Defaults;  bool SomeAttrExcluded = false;  NoScriptArgs = false;  NoCodeProperties = false;  SomeAttrIncluded = false;  #define ADD(STR, EXCEPT) \    if (FLAGCLEAR(Options, EXCEPT)) \    { \      AddToList(Result, (STR), Separator); \      SomeAttrIncluded = true; \    } \    else \    { \      SomeAttrExcluded = true; \    }  bool AsciiFileMaskDiffers = (TransferMode == tmAutomatic) && !(AsciiFileMask == Defaults.AsciiFileMask);  bool TransferModeDiffers = ((TransferMode != Defaults.TransferMode) || AsciiFileMaskDiffers);  if (FLAGCLEAR(Options, cpaIncludeMaskOnly | cpaNoTransferMode))  {    // Adding Transfer type unconditionally    bool FormatMask;    int Ident;    switch (TransferMode)    {      case tmBinary:        FormatMask = false;        Ident = 2;        break;      case tmAscii:        FormatMask = false;        Ident = 3;        break;      case tmAutomatic:      default:        FormatMask = !(AsciiFileMask == Defaults.AsciiFileMask);        Ident = FormatMask ? 4 : 5;        break;    }    UnicodeString S = FORMAT(LoadStrPart(COPY_INFO_TRANSFER_TYPE2, 1),      (LoadStrPart(COPY_INFO_TRANSFER_TYPE2, Ident)));    if (FormatMask)    {      S = FORMAT(S, (AsciiFileMask.Masks));    }    AddToList(Result, S, Separator);    if (TransferModeDiffers)    {      ADD("", cpaIncludeMaskOnly | cpaNoTransferMode);      ScriptArgs += RtfSwitchValue(TRANSFER_SWITCH, Link, TransferModeNames[TransferMode]);      const wchar_t * TransferModeMembers[] = { L"Binary", L"Ascii", L"Automatic" };      AssemblyCode += AssemblyProperty(        Language, TransferOptionsClassName, L"TransferMode", L"TransferMode", TransferModeMembers[TransferMode], false);      if (AsciiFileMaskDiffers)      {        NoScriptArgs = true;        NoCodeProperties = true;      }    }  }  else  {    if (TransferModeDiffers)    {      SomeAttrExcluded = true;      NoScriptArgs = true;      NoCodeProperties = true;    }  }  if (FileNameCase != Defaults.FileNameCase)  {    ADD(FORMAT(LoadStrPart(COPY_INFO_FILENAME, 1),      (LoadStrPart(COPY_INFO_FILENAME, FileNameCase + 2))),      cpaIncludeMaskOnly);    NoScriptArgs = true;    NoCodeProperties = true;  }  if ((InvalidCharsReplacement == NoReplacement) !=        (Defaults.InvalidCharsReplacement == NoReplacement))  {    DebugAssert(InvalidCharsReplacement == NoReplacement);    if (InvalidCharsReplacement == NoReplacement)    {      ADD(LoadStr(COPY_INFO_DONT_REPLACE_INV_CHARS), cpaIncludeMaskOnly);    }    NoScriptArgs = true;    NoCodeProperties = true;  }  if ((PreserveRights != Defaults.PreserveRights) ||      (PreserveRights &&       ((Rights != Defaults.Rights) || (AddXToDirectories != Defaults.AddXToDirectories))))  {    const int Except = cpaIncludeMaskOnly | cpaNoRights;    if (DebugAlwaysTrue(PreserveRights))    {      UnicodeString RightsStr = Rights.Text;      if (AddXToDirectories)      {        RightsStr += L", " + LoadStr(COPY_INFO_ADD_X_TO_DIRS);      }      ADD(FORMAT(LoadStr(COPY_INFO_PERMISSIONS), (RightsStr)),        Except);      if (FLAGCLEAR(Options, Except))      {        ScriptArgs += RtfSwitchValue(PERMISSIONS_SWITCH, Link, Rights.Octal);        const UnicodeString FilePermissionsClassName = L"FilePermissions";        const bool Inline = true;        UnicodeString FilePermissions =          AssemblyNewClassInstanceStart(Language, FilePermissionsClassName, Inline) +          AssemblyProperty(Language, FilePermissionsClassName, L"Octal", Rights.Octal, Inline) +          AssemblyNewClassInstanceEnd(Language, Inline);        AssemblyCode += AssemblyPropertyRaw(Language, TransferOptionsClassName, L"FilePermissions", FilePermissions, false);      }    }    if ((AddXToDirectories != Defaults.AddXToDirectories) && FLAGCLEAR(Options, Except))    {      NoScriptArgs = true;      NoCodeProperties = true;    }  }  bool AddPreserveTime = (PreserveTime != Defaults.PreserveTime);  bool APreserveTimeDirs = PreserveTime && PreserveTimeDirs;  if (AddPreserveTime || (APreserveTimeDirs != Defaults.PreserveTimeDirs))  {    UnicodeString Str = LoadStr(PreserveTime ? COPY_INFO_TIMESTAMP : COPY_INFO_DONT_PRESERVE_TIME);    const int Except = cpaIncludeMaskOnly | cpaNoPreserveTime;    const int ExceptDirs = cpaNoPreserveTimeDirs;    if (APreserveTimeDirs != Defaults.PreserveTimeDirs)    {      if (DebugAlwaysTrue(PreserveTimeDirs))      {        if (FLAGCLEAR(Options, ExceptDirs))        {          Str = FMTLOAD(COPY_INFO_PRESERVE_TIME_DIRS, (Str));          AddPreserveTime = true;        }      }      ADD("", Except | ExceptDirs);    }    if (AddPreserveTime)    {      ADD(Str, Except);    }    if (FLAGCLEAR(Options, Except))    {      if (PreserveTime)      {        if (PreserveTimeDirs && FLAGCLEAR(Options, ExceptDirs))        {          ScriptArgs += RtfSwitchValue(PRESERVETIME_SWITCH, Link, PRESERVETIMEDIRS_SWITCH_VALUE);          NoCodeProperties = true;        }        else        {          DebugFail(); // should never get here          ScriptArgs += RtfSwitch(PRESERVETIME_SWITCH, Link);        }      }      else      {        ScriptArgs += RtfSwitch(NOPRESERVETIME_SWITCH, Link);        AssemblyCode += AssemblyProperty(Language, TransferOptionsClassName, L"PreserveTimestamp", false, false);      }    }  }  if ((PreserveRights || PreserveTime) &&      (IgnorePermErrors != Defaults.IgnorePermErrors))  {    if (DebugAlwaysTrue(IgnorePermErrors))    {      const int Except = cpaIncludeMaskOnly | cpaNoIgnorePermErrors;      ADD(LoadStr(COPY_INFO_IGNORE_PERM_ERRORS), Except);      if (FLAGCLEAR(Options, Except))      {        NoScriptArgs = true;        NoCodeProperties = true;      }    }  }  if (PreserveReadOnly != Defaults.PreserveReadOnly)  {    if (DebugAlwaysTrue(PreserveReadOnly))    {      const int Except = cpaIncludeMaskOnly | cpaNoPreserveReadOnly;      ADD(LoadStr(COPY_INFO_PRESERVE_READONLY), Except);      if (FLAGCLEAR(Options, Except))      {        NoScriptArgs = true;        NoCodeProperties = true;      }    }  }  if (CalculateSize != Defaults.CalculateSize)  {    if (DebugAlwaysTrue(!CalculateSize))    {      ADD(LoadStr(COPY_INFO_DONT_CALCULATE_SIZE), cpaIncludeMaskOnly);      // Always false in scripting, in assembly controlled by use of FileTransferProgress    }  }  if (ClearArchive != Defaults.ClearArchive)  {    if (DebugAlwaysTrue(ClearArchive))    {      const int Except = cpaIncludeMaskOnly | cpaNoClearArchive;      ADD(LoadStr(COPY_INFO_CLEAR_ARCHIVE), Except);      if (FLAGCLEAR(Options, Except))      {        NoScriptArgs = true;        NoCodeProperties = true;      }    }  }  if ((TransferMode == tmAscii) || (TransferMode == tmAutomatic))  {    if (RemoveBOM != Defaults.RemoveBOM)    {      if (DebugAlwaysTrue(RemoveBOM))      {        const int Except = cpaIncludeMaskOnly | cpaNoRemoveBOM | cpaNoTransferMode;        ADD(LoadStr(COPY_INFO_REMOVE_BOM), Except);        if (FLAGCLEAR(Options, Except))        {          NoScriptArgs = true;          NoCodeProperties = true;        }      }    }    if (RemoveCtrlZ != Defaults.RemoveCtrlZ)    {      if (DebugAlwaysTrue(RemoveCtrlZ))      {        const int Except = cpaIncludeMaskOnly | cpaNoRemoveCtrlZ | cpaNoTransferMode;        ADD(LoadStr(COPY_INFO_REMOVE_CTRLZ),Except);        if (FLAGCLEAR(Options, Except))        {          NoScriptArgs = true;          NoCodeProperties = true;        }      }    }  }  if (!(IncludeFileMask == Defaults.IncludeFileMask))  {    ADD(FORMAT(LoadStr(COPY_INFO_FILE_MASK), (IncludeFileMask.Masks)),      cpaNoIncludeMask);    ScriptArgs += RtfSwitch(FILEMASK_SWITCH, Link, IncludeFileMask.Masks);    AssemblyCode += AssemblyProperty(Language, TransferOptionsClassName, L"FileMask", IncludeFileMask.Masks, false);  }  DebugAssert(FTransferSkipList.get() == NULL);  DebugAssert(FTransferResumeFile.IsEmpty());  if (CPSLimit > 0)  {    int LimitKB = int(CPSLimit / 1024);    ADD(FMTLOAD(COPY_INFO_CPS_LIMIT2, (LimitKB)), cpaIncludeMaskOnly);    ScriptArgs += RtfSwitch(SPEED_SWITCH, Link, LimitKB);    AssemblyCode += AssemblyProperty(Language, TransferOptionsClassName, L"Speed", LimitKB, false);  }  if (NewerOnly != Defaults.NewerOnly)  {    if (DebugAlwaysTrue(NewerOnly))    {      const int Except = cpaIncludeMaskOnly | cpaNoNewerOnly;      ADD(StripHotkey(LoadStr(COPY_PARAM_NEWER_ONLY)), Except);      if (FLAGCLEAR(Options, Except))      {        ScriptArgs += RtfSwitch(NEWERONLY_SWICH, Link);        NoCodeProperties = true;      }    }  }  bool ResumeThresholdDiffers = ((ResumeSupport == rsSmart) && (ResumeThreshold != Defaults.ResumeThreshold));  if (((ResumeSupport != Defaults.ResumeSupport) || ResumeThresholdDiffers) &&      (TransferMode != tmAscii) && FLAGCLEAR(Options, cpaNoResumeSupport))  {    UnicodeString Value;    UnicodeString CodeState;    int ResumeThresholdKB = (ResumeThreshold / 1024);    switch (ResumeSupport)    {      case rsOff:        Value = ToggleNames[ToggleOff];        CodeState = L"Off";        break;      case rsOn:        Value = ToggleNames[ToggleOn];        CodeState = L"On";        break;      case rsSmart:        Value = IntToStr(ResumeThresholdKB);        break;    }    ScriptArgs += RtfSwitchValue(RESUMESUPPORT_SWITCH, Link, Value);    const UnicodeString ResumeSupportClassName = L"TransferResumeSupport";    const bool Inline = true;    UnicodeString ResumeSupportCode =      AssemblyNewClassInstanceStart(Language, ResumeSupportClassName, Inline);    if (ResumeSupport == rsSmart)    {      ResumeSupportCode += AssemblyProperty(Language, ResumeSupportClassName, L"Threshold", ResumeThresholdKB, Inline);    }    else    {      ResumeSupportCode += AssemblyProperty(Language, ResumeSupportClassName, L"State", L"TransferResumeSupportState", CodeState, Inline);    }    ResumeSupportCode += AssemblyNewClassInstanceEnd(Language, Inline);    AssemblyCode += AssemblyPropertyRaw(Language, TransferOptionsClassName, L"ResumeSupport", ResumeSupportCode, false);  }  if (SomeAttrExcluded)  {    Result += (Result.IsEmpty() ? UnicodeString() : Separator) +      FORMAT(LoadStrPart(COPY_INFO_NOT_USABLE, 1),        (LoadStrPart(COPY_INFO_NOT_USABLE, (SomeAttrIncluded ? 2 : 3))));  }  else if (Result.IsEmpty())  {    Result = LoadStr(COPY_INFO_DEFAULT);  }  #undef ADD}//---------------------------------------------------------------------------void __fastcall TCopyParamType::Assign(const TCopyParamType * Source){  DebugAssert(Source != NULL);  #define COPY(Prop) Prop = Source->Prop  COPY(FileNameCase);  COPY(PreserveReadOnly);  COPY(PreserveTime);  COPY(PreserveTimeDirs);  COPY(Rights);  COPY(AsciiFileMask);  COPY(TransferMode);  COPY(AddXToDirectories);  COPY(PreserveRights);  COPY(IgnorePermErrors);  COPY(ResumeSupport);  COPY(ResumeThreshold);  COPY(InvalidCharsReplacement);  COPY(LocalInvalidChars);  COPY(CalculateSize);  COPY(FileMask);  COPY(IncludeFileMask);  COPY(TransferSkipList);  COPY(TransferResumeFile);  COPY(ClearArchive);  COPY(RemoveCtrlZ);  COPY(RemoveBOM);  COPY(CPSLimit);  COPY(NewerOnly);  #undef COPY}//---------------------------------------------------------------------------TCopyParamType & __fastcall TCopyParamType::operator =(const TCopyParamType & rhp){  Assign(&rhp);  return *this;}//---------------------------------------------------------------------------void __fastcall TCopyParamType::SetLocalInvalidChars(UnicodeString value){  if (value != LocalInvalidChars)  {    FLocalInvalidChars = value;    FTokenizibleChars = FLocalInvalidChars + TokenPrefix;  }}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::GetReplaceInvalidChars() const{  return (InvalidCharsReplacement != NoReplacement);}//---------------------------------------------------------------------------void __fastcall TCopyParamType::SetReplaceInvalidChars(bool value){  if (ReplaceInvalidChars != value)  {    InvalidCharsReplacement = (value ? TokenReplacement : NoReplacement);  }}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::ValidLocalFileName(UnicodeString FileName) const{  return ::ValidLocalFileName(FileName, InvalidCharsReplacement, FTokenizibleChars, LocalInvalidChars);}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::RestoreChars(UnicodeString FileName) const{  if (InvalidCharsReplacement == TokenReplacement)  {    wchar_t * InvalidChar = FileName.c_str();    while ((InvalidChar = wcschr(InvalidChar, TokenPrefix)) != NULL)    {      int Index = InvalidChar - FileName.c_str() + 1;      if (FileName.Length() >= Index + 2)      {        UnicodeString Hex = FileName.SubString(Index + 1, 2);        wchar_t Char = static_cast<wchar_t>(HexToByte(Hex));        if ((Char != L'\0') &&            ((FTokenizibleChars.Pos(Char) > 0) ||             (((Char == L' ') || (Char == L'.')) && (Index == FileName.Length() - 2))))        {          FileName[Index] = Char;          FileName.Delete(Index + 1, 2);          InvalidChar = FileName.c_str() + Index;        }        else if ((Hex == L"00") &&                 ((Index == FileName.Length() - 2) || (FileName[Index + 3] == L'.')) &&                 IsReservedName(FileName.SubString(1, Index - 1) + FileName.SubString(Index + 3, FileName.Length() - Index - 3 + 1)))        {          FileName.Delete(Index, 3);          InvalidChar = FileName.c_str() + Index - 1;        }        else        {          InvalidChar++;        }      }      else      {        InvalidChar++;      }    }  }  return FileName;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::ValidLocalPath(UnicodeString Path) const{  UnicodeString Result;  while (!Path.IsEmpty())  {    if (!Result.IsEmpty())    {      Result += L"\\";    }    Result += ValidLocalFileName(CutToChar(Path, L'\\', false));  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::ChangeFileName(UnicodeString FileName,  TOperationSide Side, bool FirstLevel) const{  if (FirstLevel)  {    FileName = MaskFileName(FileName, FileMask);  }  switch (FileNameCase) {    case ncUpperCase: FileName = FileName.UpperCase(); break;    case ncLowerCase: FileName = FileName.LowerCase(); break;    case ncFirstUpperCase: FileName = FileName.SubString(1, 1).UpperCase() +      FileName.SubString(2, FileName.Length()-1).LowerCase(); break;    case ncLowerCaseShort:      if ((FileName.Length() <= 12) && (FileName.Pos(L".") <= 9) &&          (FileName == FileName.UpperCase()))      {        FileName = FileName.LowerCase();      }      break;    case ncNoChange:    default:      /*nothing*/      break;  }  if (Side == osRemote)  {    FileName = ValidLocalFileName(FileName);  }  else  {    FileName = RestoreChars(FileName);  }  return FileName;}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::UseAsciiTransfer(UnicodeString FileName,  TOperationSide Side, const TFileMasks::TParams & Params) const{  switch (TransferMode)  {    case tmBinary: return false;    case tmAscii: return true;    case tmAutomatic: return AsciiFileMask.Matches(FileName, (Side == osLocal),      false, &Params);    default: DebugFail(); return false;  }}//---------------------------------------------------------------------------TRights __fastcall TCopyParamType::RemoteFileRights(Integer Attrs) const{  TRights R = Rights;  if ((Attrs & faDirectory) && AddXToDirectories)    R.AddExecute();  return R;}//---------------------------------------------------------------------------UnicodeString __fastcall TCopyParamType::GetLogStr() const{  wchar_t CaseC[] = L"NULFS";  wchar_t ModeC[] = L"BAM";  wchar_t ResumeC[] = L"YSN";  // OpenArray (ARRAYOFCONST) supports only up to 19 arguments, so we had to split it  return    FORMAT(      L"  PrTime: %s%s; PrRO: %s; Rght: %s; PrR: %s (%s); FnCs: %s; RIC: %s; "         "Resume: %s (%d); CalcS: %s; Mask: %s\n",      (BooleanToEngStr(PreserveTime),       UnicodeString(PreserveTime && PreserveTimeDirs ? L"+Dirs" : L""),       BooleanToEngStr(PreserveReadOnly),       Rights.Text,       BooleanToEngStr(PreserveRights),       BooleanToEngStr(IgnorePermErrors),       CaseC[FileNameCase],       CharToHex(InvalidCharsReplacement),       ResumeC[ResumeSupport],       (int)ResumeThreshold,       BooleanToEngStr(CalculateSize),       FileMask)) +    FORMAT(      L"  TM: %s; ClAr: %s; RemEOF: %s; RemBOM: %s; CPS: %u; NewerOnly: %s; InclM: %s; ResumeL: %d\n"       "  AscM: %s\n",      (ModeC[TransferMode],       BooleanToEngStr(ClearArchive),       BooleanToEngStr(RemoveCtrlZ),       BooleanToEngStr(RemoveBOM),       int(CPSLimit),       BooleanToEngStr(NewerOnly),       IncludeFileMask.Masks,       ((FTransferSkipList.get() != NULL) ? FTransferSkipList->Count : 0) + (!FTransferResumeFile.IsEmpty() ? 1 : 0),       AsciiFileMask.Masks));}//---------------------------------------------------------------------------int __fastcall TCopyParamType::LocalFileAttrs(const TRights & Rights) const{  int Result = 0;  if (PreserveReadOnly && !Rights.Right[TRights::rrUserWrite])  {    Result |= faReadOnly;  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::AllowResume(__int64 Size) const{  switch (ResumeSupport)  {    case rsOn: return true;    case rsOff: return false;    case rsSmart: return (Size >= ResumeThreshold);    default: DebugFail(); return false;  }}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::AllowAnyTransfer() const{  return    IncludeFileMask.Masks.IsEmpty() &&    ((FTransferSkipList.get() == NULL) || (FTransferSkipList->Count == 0)) &&    FTransferResumeFile.IsEmpty();}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::AllowTransfer(UnicodeString FileName,  TOperationSide Side, bool Directory, const TFileMasks::TParams & Params) const{  bool Result = true;  if (!IncludeFileMask.Masks.IsEmpty())  {    Result = IncludeFileMask.Matches(FileName, (Side == osLocal),      Directory, &Params);  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::SkipTransfer(  UnicodeString FileName, bool Directory) const{  bool Result = false;  // we deliberatelly do not filter directories, as path is added to resume list  // when a transfer of file or directory is started,  // so for directories we need to recurse and check every single file  if (!Directory && (FTransferSkipList.get() != NULL))  {    Result = (FTransferSkipList->IndexOf(FileName) >= 0);  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TCopyParamType::ResumeTransfer(UnicodeString FileName) const{  // Returning true has the same effect as cpResume  return    (FileName == FTransferResumeFile) &&    DebugAlwaysTrue(!FTransferResumeFile.IsEmpty());}//---------------------------------------------------------------------------TStrings * __fastcall TCopyParamType::GetTransferSkipList() const{  return FTransferSkipList.get();}//---------------------------------------------------------------------------void __fastcall TCopyParamType::SetTransferSkipList(TStrings * value){  if ((value == NULL) || (value->Count == 0))  {    FTransferSkipList.reset(NULL);  }  else  {    FTransferSkipList.reset(new TStringList());    FTransferSkipList->AddStrings(value);    FTransferSkipList->Sorted = true;  }}//---------------------------------------------------------------------------void __fastcall TCopyParamType::Load(THierarchicalStorage * Storage){  AddXToDirectories = Storage->ReadBool(L"AddXToDirectories", AddXToDirectories);  AsciiFileMask.Masks = Storage->ReadString(L"Masks", AsciiFileMask.Masks);  FileNameCase = (TFileNameCase)Storage->ReadInteger(L"FileNameCase", FileNameCase);  PreserveReadOnly = Storage->ReadBool(L"PreserveReadOnly", PreserveReadOnly);  PreserveTime = Storage->ReadBool(L"PreserveTime", PreserveTime);  PreserveTimeDirs = Storage->ReadBool(L"PreserveTimeDirs", PreserveTimeDirs);  PreserveRights = Storage->ReadBool(L"PreserveRights", PreserveRights);  IgnorePermErrors = Storage->ReadBool(L"IgnorePermErrors", IgnorePermErrors);  Rights.Text = Storage->ReadString(L"Text", Rights.Text);  TransferMode = (TTransferMode)Storage->ReadInteger(L"TransferMode", TransferMode);  ResumeSupport = (TResumeSupport)Storage->ReadInteger(L"ResumeSupport", ResumeSupport);  ResumeThreshold = Storage->ReadInt64(L"ResumeThreshold", ResumeThreshold);  InvalidCharsReplacement = (wchar_t)Storage->ReadInteger(L"ReplaceInvalidChars",    (unsigned int)InvalidCharsReplacement);  LocalInvalidChars = Storage->ReadString(L"LocalInvalidChars", LocalInvalidChars);  CalculateSize = Storage->ReadBool(L"CalculateSize", CalculateSize);  if (Storage->ValueExists(L"IncludeFileMask"))  {    IncludeFileMask.Masks = Storage->ReadString(L"IncludeFileMask", IncludeFileMask.Masks);  }  else if (Storage->ValueExists(L"ExcludeFileMask"))  {    UnicodeString ExcludeFileMask = Storage->ReadString(L"ExcludeFileMask", L"");    if (!ExcludeFileMask.IsEmpty())    {      bool NegativeExclude = Storage->ReadBool(L"NegativeExclude", false);      if (NegativeExclude)      {        IncludeFileMask.Masks = ExcludeFileMask;      }      // convert at least simple cases to new format      else if (ExcludeFileMask.Pos(IncludeExcludeFileMasksDelimiter) == 0)      {        IncludeFileMask.Masks = UnicodeString(IncludeExcludeFileMasksDelimiter) + ExcludeFileMask;      }    }  }  TransferSkipList = NULL;  TransferResumeFile = L"";  ClearArchive = Storage->ReadBool(L"ClearArchive", ClearArchive);  RemoveCtrlZ = Storage->ReadBool(L"RemoveCtrlZ", RemoveCtrlZ);  RemoveBOM = Storage->ReadBool(L"RemoveBOM", RemoveBOM);  CPSLimit = Storage->ReadInteger(L"CPSLimit", CPSLimit);  NewerOnly = Storage->ReadBool(L"NewerOnly", NewerOnly);}//---------------------------------------------------------------------------void __fastcall TCopyParamType::Save(THierarchicalStorage * Storage) const{  Storage->WriteBool(L"AddXToDirectories", AddXToDirectories);  Storage->WriteString(L"Masks", AsciiFileMask.Masks);  Storage->WriteInteger(L"FileNameCase", FileNameCase);  Storage->WriteBool(L"PreserveReadOnly", PreserveReadOnly);  Storage->WriteBool(L"PreserveTime", PreserveTime);  Storage->WriteBool(L"PreserveTimeDirs", PreserveTimeDirs);  Storage->WriteBool(L"PreserveRights", PreserveRights);  Storage->WriteBool(L"IgnorePermErrors", IgnorePermErrors);  Storage->WriteString(L"Text", Rights.Text);  Storage->WriteInteger(L"TransferMode", TransferMode);  Storage->WriteInteger(L"ResumeSupport", ResumeSupport);  Storage->WriteInt64(L"ResumeThreshold", ResumeThreshold);  Storage->WriteInteger(L"ReplaceInvalidChars", (unsigned int)InvalidCharsReplacement);  Storage->WriteString(L"LocalInvalidChars", LocalInvalidChars);  Storage->WriteBool(L"CalculateSize", CalculateSize);  Storage->WriteString(L"IncludeFileMask", IncludeFileMask.Masks);  Storage->DeleteValue(L"ExcludeFileMask"); // obsolete  Storage->DeleteValue(L"NegativeExclude"); // obsolete  DebugAssert(FTransferSkipList.get() == NULL);  DebugAssert(FTransferResumeFile.IsEmpty());  Storage->WriteBool(L"ClearArchive", ClearArchive);  Storage->WriteBool(L"RemoveCtrlZ", RemoveCtrlZ);  Storage->WriteBool(L"RemoveBOM", RemoveBOM);  Storage->WriteInteger(L"CPSLimit", CPSLimit);  Storage->WriteBool(L"NewerOnly", NewerOnly);}//---------------------------------------------------------------------------#define C(Property) (Property == rhp.Property)bool __fastcall TCopyParamType::operator==(const TCopyParamType & rhp) const{  DebugAssert(FTransferSkipList.get() == NULL);  DebugAssert(FTransferResumeFile.IsEmpty());  DebugAssert(rhp.FTransferSkipList.get() == NULL);  DebugAssert(rhp.FTransferResumeFile.IsEmpty());  return    C(AddXToDirectories) &&    C(AsciiFileMask) &&    C(FileNameCase) &&    C(PreserveReadOnly) &&    C(PreserveTime) &&    C(PreserveTimeDirs) &&    C(PreserveRights) &&    C(IgnorePermErrors) &&    C(Rights) &&    C(TransferMode) &&    C(ResumeSupport) &&    C(ResumeThreshold) &&    C(InvalidCharsReplacement) &&    C(LocalInvalidChars) &&    C(CalculateSize) &&    C(IncludeFileMask) &&    C(ClearArchive) &&    C(RemoveCtrlZ) &&    C(RemoveBOM) &&    C(CPSLimit) &&    C(NewerOnly) &&    true;}#undef C//---------------------------------------------------------------------------static bool __fastcall TryGetSpeedLimit(const UnicodeString & Text, unsigned long & Speed){  bool Result;  if (AnsiSameText(Text, LoadStr(SPEED_UNLIMITED)))  {    Speed = 0;    Result = true;  }  else  {    int SSpeed;    Result = TryStrToInt(Text, SSpeed) && (SSpeed >= 0);    if (Result)    {      Speed = SSpeed * 1024;    }  }  return Result;}//---------------------------------------------------------------------------unsigned long __fastcall GetSpeedLimit(const UnicodeString & Text){  unsigned long Speed;  if (!TryGetSpeedLimit(Text, Speed))  {    throw Exception(FMTLOAD(SPEED_INVALID, (Text)));  }  return Speed;}//---------------------------------------------------------------------------UnicodeString __fastcall SetSpeedLimit(unsigned long Limit){  UnicodeString Text;  if (Limit == 0)  {    Text = LoadStr(SPEED_UNLIMITED);  }  else  {    Text = IntToStr(int(Limit / 1024));  }  return Text;}//---------------------------------------------------------------------------void __fastcall CopySpeedLimits(TStrings * Source, TStrings * Dest){  std::unique_ptr<TStringList> Temp(new TStringList());  bool Unlimited = false;  for (int Index = 0; Index < Source->Count; Index++)  {    UnicodeString Text = Source->Strings[Index];    unsigned long Speed;    bool Valid = TryGetSpeedLimit(Text, Speed);    if ((!Valid || (Speed == 0)) && !Unlimited)    {      Temp->Add(LoadStr(SPEED_UNLIMITED));      Unlimited = true;    }    else if (Valid && (Speed > 0))    {      Temp->Add(Text);    }  }  if (!Unlimited)  {    Temp->Insert(0, LoadStr(SPEED_UNLIMITED));  }  Dest->Assign(Temp.get());}
 |