| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "Common.h"#include "WinConfiguration.h"#include "Exceptions.h"#include "Bookmarks.h"#include "Terminal.h"#include "TextsWin.h"#include "WinInterface.h"#include "GUITools.h"#include "Tools.h"#include "Setup.h"#include "Security.h"#include "TerminalManager.h"#include "Cryptography.h"#include <VCLCommon.h>#include <InitGUID.h>#include <DragExt.h>#include <Math.hpp>#include <StrUtils.hpp>#include <Generics.Defaults.hpp>#include <OperationWithTimeout.hpp>#include "FileInfo.h"//---------------------------------------------------------------------------#pragma package(smart_init)//---------------------------------------------------------------------------TWinConfiguration * WinConfiguration = NULL;//---------------------------------------------------------------------------static UnicodeString NotepadName(L"notepad.exe");static UnicodeString ToolbarsLayoutKey(L"ToolbarsLayout2");static UnicodeString ToolbarsLayoutOldKey(L"ToolbarsLayout");//---------------------------------------------------------------------------static const wchar_t FileColorDataSeparator = L':';TFileColorData::TFileColorData() :  Color(TColor()){}//---------------------------------------------------------------------------void TFileColorData::Load(const UnicodeString & S){  UnicodeString Buf(S);  Color = RestoreColor(CutToChar(Buf, FileColorDataSeparator, true));  FileMask = Buf;}//---------------------------------------------------------------------------UnicodeString TFileColorData::Save() const{  UnicodeString Result = StoreColor(Color) + FileColorDataSeparator + FileMask.Masks;  return Result;}//---------------------------------------------------------------------------void TFileColorData::LoadList(const UnicodeString & S, TList & List){  std::unique_ptr<TStringList> Strings(new TStringList());  Strings->CommaText = S;  List.clear();  for (int Index = 0; Index < Strings->Count; Index++)  {    TFileColorData FileColorData;    FileColorData.Load(Strings->Strings[Index]);    List.push_back(FileColorData);  }}//---------------------------------------------------------------------------UnicodeString TFileColorData::SaveList(const TList & List){  std::unique_ptr<TStringList> Strings(new TStringList());  for (TFileColorData::TList::const_iterator Iter = List.begin(); Iter != List.end(); Iter++)  {    Strings->Add((*Iter).Save());  }  return Strings->CommaText;}//---------------------------------------------------------------------------//---------------------------------------------------------------------------__fastcall TEditorData::TEditorData() :  FileMask(AnyMask),  Editor(edInternal),  ExternalEditor(L""),  ExternalEditorText(false),  SDIExternalEditor(false),  DetectMDIExternalEditor(false){}//---------------------------------------------------------------------------__fastcall TEditorData::TEditorData(const TEditorData & Source) :  FileMask(Source.FileMask),  Editor(Source.Editor),  ExternalEditor(Source.ExternalEditor),  ExternalEditorText(Source.ExternalEditorText),  SDIExternalEditor(Source.SDIExternalEditor),  DetectMDIExternalEditor(Source.DetectMDIExternalEditor){}//---------------------------------------------------------------------------#define C(Property) (Property == rhd.Property)bool __fastcall TEditorData::operator==(const TEditorData & rhd) const{  return    C(FileMask) &&    C(Editor) &&    C(ExternalEditor) &&    C(ExternalEditorText) &&    C(SDIExternalEditor) &&    C(DetectMDIExternalEditor) &&    true;}#undef C//---------------------------------------------------------------------------bool __fastcall TEditorData::DecideExternalEditorText(UnicodeString ExternalEditor){  bool Result = false;  // By default we use default transfer mode (binary),  // as all reasonable 3rd party editors support all EOL styles.  // A notable exception is Windows Notepad, so here's an exception for it.  ReformatFileNameCommand(ExternalEditor);  UnicodeString ProgramName = ExtractProgramName(ExternalEditor);  // We explicitly do not use TEditorPreferences::GetDefaultExternalEditor(),  // as we need to explicitly refer to the Notepad, even if the default external  // editor ever changes  UnicodeString NotepadProgramName = ExtractProgramName(NotepadName);  if (SameText(ProgramName, NotepadProgramName))  {    Result = true;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TEditorData::DecideExternalEditorText(){  if (DecideExternalEditorText(ExternalEditor))  {    ExternalEditorText = true;  }}//---------------------------------------------------------------------------__fastcall TEditorPreferences::TEditorPreferences(){}//---------------------------------------------------------------------------__fastcall TEditorPreferences::TEditorPreferences(const TEditorData & Data) :  FData(Data){}//---------------------------------------------------------------------------bool __fastcall TEditorPreferences::operator==(const TEditorPreferences & rhp) const{  return (FData == rhp.FData);}#undef C//---------------------------------------------------------------------------bool __fastcall TEditorPreferences::Matches(const UnicodeString FileName,  bool Local, const TFileMasks::TParams & Params) const{  return FData.FileMask.Matches(FileName, Local, false, &Params);}//---------------------------------------------------------------------------UnicodeString __fastcall TEditorPreferences::GetDefaultExternalEditor(){  return NotepadName;}//---------------------------------------------------------------------------void __fastcall TEditorPreferences::LegacyDefaults(){  FData.ExternalEditor = GetDefaultExternalEditor();  FData.DecideExternalEditorText();}//---------------------------------------------------------------------------UnicodeString __fastcall TEditorPreferences::ExtractExternalEditorName() const{  DebugAssert(FData.Editor == edExternal);  UnicodeString ExternalEditor = FData.ExternalEditor;  ReformatFileNameCommand(ExternalEditor);  // Trim is a workaround for unknown problem with "notepad  " (2 trailing spaces)  return ExtractProgramName(ExternalEditor).Trim();}//---------------------------------------------------------------------------void __fastcall TEditorPreferences::Load(THierarchicalStorage * Storage, bool Legacy){  if (!Legacy)  {    FData.FileMask = Storage->ReadString(L"FileMask", FData.FileMask.Masks);  }  FData.Editor = (TEditor)Storage->ReadInteger(L"Editor", FData.Editor);  FData.ExternalEditor = Storage->ReadString(L"ExternalEditor", FData.ExternalEditor);  FData.ExternalEditorText = Storage->ReadBool(L"ExternalEditorText", FData.ExternalEditorText);  FData.SDIExternalEditor = Storage->ReadBool(L"SDIExternalEditor", FData.SDIExternalEditor);  FData.DetectMDIExternalEditor = Storage->ReadBool(L"DetectMDIExternalEditor", FData.DetectMDIExternalEditor);}//---------------------------------------------------------------------------void __fastcall TEditorPreferences::Save(THierarchicalStorage * Storage) const{  Storage->WriteString(L"FileMask", FData.FileMask.Masks);  Storage->WriteInteger(L"Editor", FData.Editor);  Storage->WriteString(L"ExternalEditor", FData.ExternalEditor);  Storage->WriteBool(L"ExternalEditorText", FData.ExternalEditorText);  Storage->WriteBool(L"SDIExternalEditor", FData.SDIExternalEditor);  Storage->WriteBool(L"DetectMDIExternalEditor", FData.DetectMDIExternalEditor);}//---------------------------------------------------------------------------TEditorData * __fastcall TEditorPreferences::GetData(){  // returning non-const data, possible data change, invalidate cached name  FName = L"";  return &FData;};//---------------------------------------------------------------------------UnicodeString __fastcall TEditorPreferences::GetName() const{  if (FName.IsEmpty())  {    if (FData.Editor == edInternal)    {      // StripHotkey is relic from times when INTERNAL_EDITOR_NAME was used      // also for the menu item caption      FName = StripHotkey(LoadStr(INTERNAL_EDITOR_NAME));    }    else if (FData.Editor == edOpen)    {      FName = StripHotkey(LoadStr(OPEN_EDITOR_NAME));    }    else    {      UnicodeString Program, Params, Dir;      UnicodeString ExternalEditor = FData.ExternalEditor;      ReformatFileNameCommand(ExternalEditor);      SplitCommand(ExternalEditor, Program, Params, Dir);      FName = ExtractFileName(Program);      int P = FName.LastDelimiter(L".");      if (P > 0)      {        FName.SetLength(P - 1);      }      if (FName.ByteType(1) == mbSingleByte)      {        if (FName.UpperCase() == FName)        {          FName = FName.LowerCase();        }        if (FName.LowerCase() == FName)        {          FName = FName.SubString(1, 1).UpperCase() +            FName.SubString(2, FName.Length() - 1);        }      }    }  }  return FName;}//---------------------------------------------------------------------------//---------------------------------------------------------------------------__fastcall TEditorList::TEditorList(){  Init();}//---------------------------------------------------------------------------void __fastcall TEditorList::Init(){  FEditors = new TList();  FModified = false;}//---------------------------------------------------------------------------__fastcall TEditorList::~TEditorList(){  Clear();  delete FEditors;}//---------------------------------------------------------------------void __fastcall TEditorList::Modify(){  FModified = true;}//---------------------------------------------------------------------------void __fastcall TEditorList::Saved(){  FModified = false;}//---------------------------------------------------------------------------TEditorList & __fastcall TEditorList::operator=(const TEditorList & rhl){  Clear();  for (int Index = 0; Index < rhl.Count; Index++)  {    Add(new TEditorPreferences(*rhl.Editors[Index]));  }  // there should be comparison of with the assigned list, but we rely on caller  // to do it instead (TWinConfiguration::SetEditorList)  Modify();  return *this;}//---------------------------------------------------------------------------bool __fastcall TEditorList::operator==(const TEditorList & rhl) const{  bool Result = (Count == rhl.Count);  if (Result)  {    int i = 0;    while ((i < Count) && Result)    {      Result = (*Editors[i]) == (*rhl.Editors[i]);      i++;    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TEditorList::Clear(){  for (int i = 0; i < Count; i++)  {    delete Editors[i];  }  FEditors->Clear();}//---------------------------------------------------------------------------void __fastcall TEditorList::Add(TEditorPreferences * Editor){  Insert(Count, Editor);}//---------------------------------------------------------------------------void __fastcall TEditorList::Insert(int Index, TEditorPreferences * Editor){  FEditors->Insert(Index, reinterpret_cast<TObject *>(Editor));  Modify();}//---------------------------------------------------------------------------void __fastcall TEditorList::Change(int Index, TEditorPreferences * Editor){  if (!((*Editors[Index]) == *Editor))  {    delete Editors[Index];    FEditors->Items[Index] = (reinterpret_cast<TObject *>(Editor));    Modify();  }  else  {    delete Editor;  }}//---------------------------------------------------------------------------void __fastcall TEditorList::Move(int CurIndex, int NewIndex){  if (CurIndex != NewIndex)  {    FEditors->Move(CurIndex, NewIndex);    Modify();  }}//---------------------------------------------------------------------------void __fastcall TEditorList::Delete(int Index){  DebugAssert((Index >= 0) && (Index < Count));  delete Editors[Index];  FEditors->Delete(Index);  Modify();}//---------------------------------------------------------------------------const TEditorPreferences * __fastcall TEditorList::Find(  const UnicodeString FileName, bool Local, const TFileMasks::TParams & Params) const{  const TEditorPreferences * Result = NULL;  int i = 0;  while ((i < FEditors->Count) && (Result == NULL))  {    Result = Editors[i];    if (!Result->Matches(FileName, Local, Params))    {      Result = NULL;    }    i++;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TEditorList::Load(THierarchicalStorage * Storage){  int Index = 0;  bool Next;  do  {    UnicodeString Name = IntToStr(Index);    TEditorPreferences * Editor = NULL;    try    {      Next = Storage->OpenSubKey(Name, false);      if (Next)      {        try        {          Editor = new TEditorPreferences();          Editor->Load(Storage, false);        }        __finally        {          Storage->CloseSubKey();        }      }    }    catch(...)    {      delete Editor;      throw;    }    if (Editor != NULL)    {      FEditors->Add(reinterpret_cast<TObject *>(Editor));    }    Index++;  }  while (Next);  FModified = false;}//---------------------------------------------------------------------------void __fastcall TEditorList::Save(THierarchicalStorage * Storage) const{  Storage->ClearSubKeys();  for (int Index = 0; Index < Count; Index++)  {    if (Storage->OpenSubKey(IntToStr(Index), true))    {      try      {        Editors[Index]->Save(Storage);      }      __finally      {        Storage->CloseSubKey();      }    }  }}//---------------------------------------------------------------------------int __fastcall TEditorList::GetCount() const{  int X = FEditors->Count;  return X;}//---------------------------------------------------------------------------const TEditorPreferences * __fastcall TEditorList::GetEditor(int Index) const{  return reinterpret_cast<TEditorPreferences *>(FEditors->Items[Index]);}//---------------------------------------------------------------------------bool __fastcall TEditorList::IsDefaultList() const{  bool Result = true;  for (int Index = 0; Result && (Index < Count); Index++)  {    const TEditorPreferences * Editor = GetEditor(Index);    if (Editor->Data->Editor == edInternal)    {      // noop (keeps Result true)    }    else if (Editor->Data->Editor == edExternal)    {      UnicodeString ExternalEditor = Editor->ExtractExternalEditorName();      UnicodeString DefaultExternalEditor = ExtractProgramName(TEditorPreferences::GetDefaultExternalEditor());      Result = SameText(ExternalEditor, DefaultExternalEditor);    }    else    {      Result = false;    }  }  return Result;}//---------------------------------------------------------------------------//---------------------------------------------------------------------------__fastcall TWinConfiguration::TWinConfiguration(): TCustomWinConfiguration(){  ResetSysDarkTheme();  FInvalidDefaultTranslationMessage = L"";  FDDExtInstalled = -1;  FBookmarks = new TBookmarks();  FCustomCommandList = new TCustomCommandList();  FExtensionList = new TCustomCommandList();  FEditorList = new TEditorList();  FDefaultUpdatesPeriod = 0;  FDontDecryptPasswords = 0;  FMasterPasswordSession = 0;  FMasterPasswordSessionAsked = false;  FCustomCommandOptions.reset(new TStringList());  FCustomCommandOptionsModified = false;  FExtensionTranslations.reset(new TStringList());  Default();  // This matters only if the translations are in the executable folder and auto-loaded by VCL (System.Pas - DelayLoadResourceModule)  try  {    UnicodeString ResourceModuleName = GetResourceModuleName(Application->Name, ModuleFileName().c_str());    CheckTranslationVersion(ResourceModuleName, true);  }  catch(Exception & E)  {    FInvalidDefaultTranslationMessage = E.Message;  }  // Load complete locale according to the UI language  SetLocaleInternal(NULL, true, true);  FDefaultLocale = AppliedLocale;}//---------------------------------------------------------------------------__fastcall TWinConfiguration::~TWinConfiguration(){  if (!FTemporarySessionFile.IsEmpty()) DeleteFile(ApiPath(FTemporarySessionFile));  ClearTemporaryLoginData();  ReleaseExtensionTranslations();  delete FBookmarks;  delete FCustomCommandList;  delete FExtensionList;  delete FEditorList;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::Default(){  FCustomCommandsDefaults = true;  FCustomCommandOptionsModified = false;  TCustomWinConfiguration::Default();  int WorkAreaWidthScaled = DimensionToDefaultPixelsPerInch(Screen->WorkAreaWidth);  int WorkAreaHeightScaled = DimensionToDefaultPixelsPerInch(Screen->WorkAreaHeight);  UnicodeString PixelsPerInchToolbarValue = "PixelsPerInch=" + SaveDefaultPixelsPerInch();  FDDTransferConfirmation = asAuto;  FDDTemporaryDirectory = L"";  FDDDrives = L"";  FDDWarnLackOfTempSpace = true;  FDDWarnLackOfTempSpaceRatio = 1.1;  FDDFakeFile = true;  FDDExtTimeout = MSecsPerSec;  FDeleteToRecycleBin = true;  FSelectDirectories = false;  FSelectMask = AnyMask;  FShowHiddenFiles = false;  FFormatSizeBytes = fbKilobytes;  FShowInaccesibleDirectories = true;  FConfirmTransferring = true;  FConfirmDeleting = true;  FConfirmRecycling = true;  FConfirmClosingSession = true;  FDoubleClickAction = dcaEdit;  FCopyOnDoubleClickConfirmation = false;  FDimmHiddenFiles = true;  FRenameWholeName = false;  FAutoStartSession = L"";  FExpertMode = true;  FUseLocationProfiles = false;  FUseSharedBookmarks = false;  FDefaultDirIsHome = true;  FDDDeleteDelay = 120;  FTemporaryDirectoryAppendSession = false;  FTemporaryDirectoryDeterministic = false;  FTemporaryDirectoryAppendPath = true;  FTemporaryDirectoryCleanup = true;  FConfirmTemporaryDirectoryCleanup = true;  FPreservePanelState = true;  FDarkTheme = asAuto;  FLastStoredSession = L"";  // deliberately not being saved, so that when saving ad-hoc workspace,  // we do not offer to overwrite the last saved workspace, what may be undesirable  FLastWorkspace = L"";  FAutoSaveWorkspace = false;  FAutoSaveWorkspacePasswords = false;  FAutoWorkspace = L"";  FPathInCaption = picShort;  FMinimizeToTray = false;  FMinimizeToTrayOnce = false;  FBalloonNotifications = true;  FNotificationsTimeout = 10;  FNotificationsStickTime = 2;  FCopyParamAutoSelectNotice = true;  FLockToolbars = false;  FSelectiveToolbarText = true;  FAutoOpenInPutty = false;  FRefreshRemotePanel = false;  FRefreshRemotePanelInterval = TDateTime(0, 1, 0, 0);  FPanelFont.FontName = L"";  FPanelFont.FontSize = 0;  FPanelFont.FontStyle = 0;  FPanelFont.FontCharset = DEFAULT_CHARSET;  FNaturalOrderNumericalSorting = true;  FFullRowSelect = false;  FOfferedEditorAutoConfig = false;  FVersionHistory = L"";  AddVersionToHistory();  FUseMasterPassword = false;  FPlainMasterPasswordEncrypt = L"";  FPlainMasterPasswordDecrypt = L"";  FMasterPasswordVerifier = L"";  FOpenedStoredSessionFolders = L"";  FAutoImportedFromPuttyOrFilezilla = false;  FGenerateUrlComponents = -1;  FGenerateUrlCodeTarget = guctUrl;  FGenerateUrlScriptFormat = sfScriptFile;  FGenerateUrlAssemblyLanguage = alCSharp;  FExternalSessionInExistingInstance = true;  FKeepOpenWhenNoSession = false;  FLocalIconsByExt = false;  FBidiModeOverride = lfoLanguageIfRecommended;  FFlipChildrenOverride = lfoLanguageIfRecommended;  FShowTips = true;  FTipsSeen = L"";  FTipsShown = Now();  FFileColors = L"";  FRunsSinceLastTip = 0;  FExtensionsDeleted = L"";  FLockedInterface = false;  HonorDrivePolicy = true;  TimeoutShellOperations = true;  TimeoutShellIconRetrieval = false;  FEditor.Font.FontName = DefaultFixedWidthFontName;  FEditor.Font.FontSize = DefaultFixedWidthFontSize;  FEditor.Font.FontStyle = 0;  FEditor.Font.FontCharset = DEFAULT_CHARSET;  FEditor.FontColor = TColor(0);  FEditor.BackgroundColor = TColor(0);  FEditor.WordWrap = false;  FEditor.FindText = L"";  FEditor.ReplaceText = L"";  FEditor.FindMatchCase = false;  FEditor.FindWholeWord = false;  FEditor.FindDown = true;  FEditor.TabSize = 8;  FEditor.MaxEditors = 500;  FEditor.EarlyClose = 2; // seconds  FEditor.SDIShellEditor = false;  FEditor.WindowParams = L"";  FEditor.Encoding = CP_ACP;  FEditor.WarnOnEncodingFallback = true;  FEditor.WarnOrLargeFileSize = true;  FQueueView.Height = 140;  FQueueView.HeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  // with 1000 pixels wide screen, both interfaces are wide enough to fit wider queue  FQueueView.Layout =    UnicodeString((WorkAreaWidthScaled > 1000) ? L"70,250,250,80,80,80,100" : L"70,160,160,80,80,80,100") +    // WORKAROUND (the comma), see GetListViewStr    L",;" + SaveDefaultPixelsPerInch();  FQueueView.Show = qvHideWhenEmpty;  FQueueView.LastHideShow = qvHideWhenEmpty;  FQueueView.ToolBar = true;  FQueueView.Label = true;  FEnableQueueByDefault = true;  FUpdates.Period = FDefaultUpdatesPeriod;  FUpdates.LastCheck = 0;  FUpdates.HaveResults = false;  FUpdates.ShownResults = false;  FUpdates.BetaVersions = asAuto;  FUpdates.ShowOnStartup = true;  FUpdates.AuthenticationEmail = L"";  // for backward compatibility the default is decided based on value of ProxyHost  FUpdates.ConnectionType = (TConnectionType)-1;  FUpdates.ProxyHost = L""; // keep empty (see above)  FUpdates.ProxyPort = 8080;  FUpdates.Results.Reset();  FUpdates.DotNetVersion = L"";  FUpdates.ConsoleVersion = L"";  int ExplorerWidth = Min(WorkAreaWidthScaled - 40, 960);  int ExplorerHeight = Min(WorkAreaHeightScaled - 30, 720);  FScpExplorer.WindowParams = FormatDefaultWindowParams(ExplorerWidth, ExplorerHeight);  // WORKAROUND (the semicolon, see TCustomListViewColProperties.GetParamsStr, and see other instances below)  FScpExplorer.DirViewParams = L"0;1;0|150,1;70,1;150,1;79,1;62,1;55,0;20,0;150,0;125,0;@" + SaveDefaultPixelsPerInch() + L"|6;7;8;0;1;2;3;4;5";  FScpExplorer.ToolbarsLayout =    UnicodeString(      L"Queue=1::0+-1,"       "Menu=1:TopDock:0+0,"       "Buttons=1:TopDock:2+0,"       "Selection=0:TopDock:3+0,"       "Session=0:TopDock:6+0,"       "Preferences=1:TopDock:4+0,"       "Sort=0:TopDock:5+0,"       "Address=1:TopDock:1+0,"       "Updates=1:TopDock:4+416,"       "Transfer=1:TopDock:4+194,"       "CustomCommands=0:TopDock:7+0,") +    PixelsPerInchToolbarValue;  FScpExplorer.ToolbarsButtons = UnicodeString();  FScpExplorer.SessionsTabs = true;  FScpExplorer.StatusBar = true;  FScpExplorer.LastLocalTargetDirectory = GetPersonalFolder();  FScpExplorer.ViewStyle = 0; /* vsIcon */  FScpExplorer.ShowFullAddress = true;  FScpExplorer.DriveView = true;  FScpExplorer.DriveViewWidth = 180;  FScpExplorer.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  int CommanderWidth = Min(WorkAreaWidthScaled - 40, 1090);  int CommanderHeight = Min(WorkAreaHeightScaled - 30, 700);  FScpCommander.WindowParams = FormatDefaultWindowParams(CommanderWidth, CommanderHeight);  FScpCommander.LocalPanelWidth = 0.5;  FScpCommander.SwappedPanels = false;  FScpCommander.SessionsTabs = true;  FScpCommander.StatusBar = true;  FScpCommander.NortonLikeMode = nlKeyboard;  FScpCommander.PreserveLocalDirectory = false;  FScpCommander.ToolbarsLayout =    UnicodeString(      L"Queue=1::0+-1,"       "Menu=1:TopDock:0+0,"       "Preferences=1:TopDock:1+228,"       "Session=0:TopDock:1+602,"       "Sort=0:TopDock:2+0,"       "Commands=1:TopDock:1+0,"       "Updates=1:TopDock:1+619,"       "Transfer=1:TopDock:1+364,"       "CustomCommands=0:TopDock:3+0,"       "RemoteHistory=1:RemoteTopDock:0+172,"       "RemoteNavigation=1:RemoteTopDock:0+252,"       "RemotePath=1:RemoteTopDock:0+0,"       "RemoteFile=1:RemoteTopDock:1+0,"       "RemoteSelection=1:RemoteTopDock:1+345,"       "LocalHistory=1:LocalTopDock:0+207,"       "LocalNavigation=1:LocalTopDock:0+287,"       "LocalPath=1:LocalTopDock:0+0,"       "LocalFile=1:LocalTopDock:1+0,"       "LocalSelection=1:LocalTopDock:1+329,"       "Toolbar2=0:BottomDock:1+0,"       "CommandLine=0:BottomDock:0+0,") +    PixelsPerInchToolbarValue;  FScpCommander.ToolbarsButtons = UnicodeString();  FScpCommander.CurrentPanel = osLocal;  FScpCommander.CompareByTime = true;  FScpCommander.CompareBySize = false;  FScpCommander.TreeOnLeft = false;  FScpCommander.ExplorerKeyboardShortcuts = false;  FScpCommander.SystemContextMenu = false;  FScpCommander.RemotePanel.DirViewParams = L"0;1;0|150,1;70,1;150,1;79,1;62,1;55,0;20,0;150,0;125,0;@" + SaveDefaultPixelsPerInch() + L"|6;7;8;0;1;2;3;4;5";  FScpCommander.RemotePanel.StatusBar = true;  FScpCommander.RemotePanel.DriveView = false;  FScpCommander.RemotePanel.DriveViewHeight = 100;  FScpCommander.RemotePanel.DriveViewHeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  FScpCommander.RemotePanel.DriveViewWidth = 100;  FScpCommander.RemotePanel.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  FScpCommander.LocalPanel.DirViewParams = L"0;1;0|150,1;70,1;120,1;150,1;55,0;55,0;@" + SaveDefaultPixelsPerInch() + L"|5;0;1;2;3;4";  FScpCommander.LocalPanel.StatusBar = true;  FScpCommander.LocalPanel.DriveView = false;  FScpCommander.LocalPanel.DriveViewHeight = 100;  FScpCommander.LocalPanel.DriveViewHeightPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  FScpCommander.LocalPanel.DriveViewWidth = 100;  FScpCommander.LocalPanel.DriveViewWidthPixelsPerInch = USER_DEFAULT_SCREEN_DPI;  FBookmarks->Clear();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::DefaultLocalized(){  TGUIConfiguration::DefaultLocalized();  if (FCustomCommandsDefaults)  {    FCustomCommandList->Clear();    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_EXECUTE), L"\"./!\"", 0);    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_TOUCH), L"touch \"!\"", ccApplyToDirectories | ccRecursive);    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_TAR),      FORMAT(L"tar -cz  -f \"!?%s?archive.tgz!\" !&",        (LoadStr(CUSTOM_COMMAND_TAR_ARCHIVE))), ccApplyToDirectories);    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_UNTAR),      FORMAT(L"tar -xz --directory=\"!?%s?.!\" -f \"!\"",        (LoadStr(CUSTOM_COMMAND_UNTAR_DIRECTORY))), 0);    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_GREP),      FORMAT(L"grep \"!?%s?!\" !&", (LoadStr(CUSTOM_COMMAND_GREP_PATTERN))),      ccShowResults);    FCustomCommandList->Add(LoadStr(CUSTOM_COMMAND_PRINT), L"notepad.exe /p \"!\"", ccLocal);    FCustomCommandList->Reset();    FCustomCommandsDefaults = true;  }}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::DetectRegistryStorage(HKEY RootKey){  bool Result = false;  TRegistryStorage * Storage = new TRegistryStorage(RegistryStorageKey, RootKey);  try  {    if (Storage->OpenRootKey(false))    {      Result = true;      Storage->CloseSubKey();    }  }  __finally  {    delete Storage;  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::CanWriteToStorage(){  bool Result = false;  try  {    THierarchicalStorage * Storage = CreateConfigStorage();    try    {      Storage->AccessMode = smReadWrite;      // This is actually not very good test, as we end up potentially with      // the very same config, and TIniFileStorage file won't even try to      // write the file then. Luckily, we use this for empty config only,      // so we end up with at least an empty section.      if (Storage->OpenSubKey(ConfigurationSubKey, true))      {        Storage->WriteBool(L"WriteTest", true);        Storage->DeleteValue(L"WriteTest");      }      Storage->Flush();    }    __finally    {      delete Storage;    }    Result = true;  }  catch(...)  {  }  return Result;}//---------------------------------------------------------------------------TStorage __fastcall TWinConfiguration::GetStorage(){  if (FStorage == stDetect)  {    if (FindResourceEx(NULL, RT_RCDATA, L"WINSCP_SESSION",      MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL)))    {      FTemporarySessionFile =        IncludeTrailingBackslash(SystemTemporaryDirectory()) + L"winscps.tmp";      DumpResourceToFile(L"WINSCP_SESSION", FTemporarySessionFile);      FEmbeddedSessions = true;      FTemporaryKeyFile =        IncludeTrailingBackslash(SystemTemporaryDirectory()) + L"winscpk.tmp";      if (!DumpResourceToFile(L"WINSCP_KEY", FTemporaryKeyFile))      {        FTemporaryKeyFile = L"";      }    }    FStorage = stIniFile;    if (!FileExists(ApiPath(IniFileStorageNameForReading)))    {      if (DetectRegistryStorage(HKEY_CURRENT_USER) ||          DetectRegistryStorage(HKEY_LOCAL_MACHINE) ||          // FStorage is now stIniFile, so this tests writing to an INI file.          // As we fall back to user profile folder, when application folder          // is not writtable, it is actually unlikely that the below test ever fails.          !CanWriteToStorage())      {        FStorage = stRegistry;      }    }  }  TStorage Result = TCustomWinConfiguration::GetStorage();  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::Saved(){  TCustomWinConfiguration::Saved();  FBookmarks->ModifyAll(false);  FCustomCommandList->Reset();  FEditorList->Saved();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::RecryptPasswords(TStrings * RecryptPasswordErrors){  TCustomWinConfiguration::RecryptPasswords(RecryptPasswordErrors);  try  {    TTerminalManager * Manager = TTerminalManager::Instance(false);    DebugAssert(Manager != NULL);    if (Manager != NULL)    {      Manager->RecryptPasswords();    }  }  catch (Exception & E)  {    UnicodeString Message;    if (ExceptionMessage(&E, Message))    {      // we do not expect this really to happen,      // so we do not bother providing context      RecryptPasswordErrors->Add(Message);    }  }}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetUseMasterPassword(){  return FUseMasterPassword;}//---------------------------------------------------------------------------THierarchicalStorage * TWinConfiguration::CreateScpStorage(bool & SessionList){  // Detect embedded session, if not checked yet  GetStorage();  THierarchicalStorage * Result;  if (SessionList && !FTemporarySessionFile.IsEmpty())  {    Result = TIniFileStorage::CreateFromPath(FTemporarySessionFile);    // This is session-list specific store, so the only instance,    // we do not reset the SessionList argument    // (compare TConfiguration::CreateScpStorage)  }  else  {    Result = TCustomWinConfiguration::CreateScpStorage(SessionList);  }  return Result;}//---------------------------------------------------------------------------// duplicated from core\configuration.cpp#define LASTELEM(ELEM) \  ELEM.SubString(ELEM.LastDelimiter(L".>")+1, ELEM.Length() - ELEM.LastDelimiter(L".>"))#define BLOCK(KEY, CANCREATE, BLOCK) \  if (Storage->OpenSubKey(KEY, CANCREATE, true)) try { BLOCK } __finally { Storage->CloseSubKey(); }#define KEY(TYPE, VAR) KEYEX(TYPE, VAR, PropertyToKey(TEXT(#VAR)))#define REGCONFIG(CANCREATE) \  BLOCK(L"Interface", CANCREATE, \    KEYEX(Integer,DoubleClickAction, L"CopyOnDoubleClick"); \    KEY(Bool,     CopyOnDoubleClickConfirmation); \    KEYEX(Integer, DDTransferConfirmation, L"DDTransferConfirmation2"); \    KEY(String,   DDTemporaryDirectory); \    KEY(String,   DDDrives); \    KEY(Bool,     DDWarnLackOfTempSpace); \    KEY(Float,    DDWarnLackOfTempSpaceRatio); \    KEY(Bool,     DeleteToRecycleBin); \    KEY(Bool,     DimmHiddenFiles); \    KEY(Bool,     RenameWholeName); \    KEY(Bool,     SelectDirectories); \    KEY(String,   SelectMask); \    KEY(Bool,     ShowHiddenFiles); \    KEY(Integer,  FormatSizeBytes); \    KEY(Bool,     ShowInaccesibleDirectories); \    KEY(Bool,     ConfirmTransferring); \    KEY(Bool,     ConfirmDeleting); \    KEY(Bool,     ConfirmRecycling); \    KEY(Bool,     ConfirmClosingSession); \    KEY(String,   AutoStartSession); \    KEY(Bool,     UseLocationProfiles); \    KEY(Bool,     UseSharedBookmarks); \    KEY(Integer,  LocaleSafe); \    KEY(Bool,     DDFakeFile); \    KEY(Integer,  DDExtTimeout); \    KEY(Bool,     DefaultDirIsHome); \    KEY(Bool,     TemporaryDirectoryAppendSession); \    KEY(Bool,     TemporaryDirectoryAppendPath); \    KEY(Bool,     TemporaryDirectoryDeterministic); \    KEY(Bool,     TemporaryDirectoryCleanup); \    KEY(Bool,     ConfirmTemporaryDirectoryCleanup); \    KEY(Bool,     PreservePanelState); \    KEY(Integer,  DarkTheme); \    KEY(String,   LastStoredSession); \    KEY(Bool,     AutoSaveWorkspace); \    KEY(Bool,     AutoSaveWorkspacePasswords); \    KEY(String,   AutoWorkspace); \    KEY(Integer,  PathInCaption); \    KEY(Bool,     MinimizeToTray); \    KEY(Bool,     BalloonNotifications); \    KEY(Integer,  NotificationsTimeout); \    KEY(Integer,  NotificationsStickTime); \    KEY(Bool,     CopyParamAutoSelectNotice); \    KEY(Bool,     LockToolbars); \    KEY(Bool,     SelectiveToolbarText); \    KEY(Bool,     AutoOpenInPutty); \    KEY(Bool,     RefreshRemotePanel); \    KEY(DateTime, RefreshRemotePanelInterval); \    KEYEX(String, PanelFont.FontName, L"PanelFontName"); \    KEYEX(Integer,PanelFont.FontSize, L"PanelFontSize"); \    KEYEX(Integer,PanelFont.FontStyle, L"PanelFontStyle"); \    KEYEX(Integer,PanelFont.FontCharset, L"PanelFontCharset"); \    KEY(Bool,     NaturalOrderNumericalSorting); \    KEY(Bool,     FullRowSelect); \    KEY(Bool,     OfferedEditorAutoConfig); \    KEY(Integer,  LastMonitor); \    KEY(String,   VersionHistory); \    KEY(Bool,     EnableQueueByDefault); \    KEY(String,   OpenedStoredSessionFolders); \    KEY(Bool,     AutoImportedFromPuttyOrFilezilla); \    KEY(Integer,  GenerateUrlComponents); \    KEY(Integer,  GenerateUrlCodeTarget); \    KEY(Integer,  GenerateUrlScriptFormat); \    KEY(Integer,  GenerateUrlAssemblyLanguage); \    KEY(Bool,     ExternalSessionInExistingInstance); \    KEY(Bool,     KeepOpenWhenNoSession); \    KEY(Bool,     LocalIconsByExt); \    KEY(Integer,  BidiModeOverride); \    KEY(Integer,  FlipChildrenOverride); \    KEY(Bool,     ShowTips); \    KEY(String,   TipsSeen); \    KEY(DateTime, TipsShown); \    KEY(String,   FileColors); \    KEY(Integer,  RunsSinceLastTip); \    KEY(Bool,     HonorDrivePolicy); \    KEY(Integer,  LastMachineInstallations); \    KEYEX(String, FExtensionsDeleted, L"ExtensionsDeleted"); \    KEYEX(String, FExtensionsOrder, L"ExtensionsOrder"); \    KEYEX(String, FExtensionsShortCuts, L"ExtensionsShortCuts"); \    KEY(Bool,     TimeoutShellOperations); \    KEY(Bool,     TimeoutShellIconRetrieval); \  ); \  BLOCK(L"Interface\\Editor", CANCREATE, \    KEYEX(String,   Editor.Font.FontName, L"FontName2"); \    KEY(Integer,  Editor.Font.FontSize); \    KEY(Integer,  Editor.Font.FontStyle); \    KEY(Integer,  Editor.Font.FontCharset); \    KEY(Integer,  Editor.FontColor); \    KEY(Integer,  Editor.BackgroundColor); \    KEY(Bool,     Editor.WordWrap); \    KEY(String,   Editor.FindText); \    KEY(String,   Editor.ReplaceText); \    KEY(Bool,     Editor.FindMatchCase); \    KEY(Bool,     Editor.FindWholeWord); \    KEY(Bool,     Editor.FindDown); \    KEY(Integer,  Editor.TabSize); \    KEY(Integer,  Editor.MaxEditors); \    KEY(Integer,  Editor.EarlyClose); \    KEY(Bool,     Editor.SDIShellEditor); \    KEY(String,   Editor.WindowParams); \    KEY(Integer,  Editor.Encoding); \    KEY(Bool,     Editor.WarnOnEncodingFallback); \    KEY(Bool,     Editor.WarnOrLargeFileSize); \  ); \  BLOCK(L"Interface\\QueueView", CANCREATE, \    KEY(Integer,  QueueView.Height); \    KEY(Integer,  QueueView.HeightPixelsPerInch); \    KEY(String,   QueueView.Layout); \    KEY(Integer,  QueueView.Show); \    KEY(Integer,  QueueView.LastHideShow); \    KEY(Bool,     QueueView.ToolBar); \    KEY(Bool,     QueueView.Label); \  ); \  BLOCK(L"Interface\\Updates", CANCREATE, \    KEY(Integer,  FUpdates.Period); \    KEY(DateTime, FUpdates.LastCheck); \    KEY(Integer,  FUpdates.HaveResults); \    KEY(Integer,  FUpdates.ShownResults); \    KEY(Integer,  FUpdates.BetaVersions); \    KEY(Bool,     FUpdates.ShowOnStartup); \    KEY(String,   FUpdates.AuthenticationEmail); \    KEY(Integer,  FUpdates.ConnectionType); \    KEY(String,   FUpdates.ProxyHost); \    KEY(Integer,  FUpdates.ProxyPort); \    KEY(Integer,  FUpdates.Results.ForVersion); \    KEY(Integer,  FUpdates.Results.Version); \    KEY(String,   FUpdates.Results.Message); \    KEY(Integer,  FUpdates.Results.Critical); \    KEY(String,   FUpdates.Results.Release); \    KEY(Bool,     FUpdates.Results.Disabled); \    KEY(String,   FUpdates.Results.Url); \    KEY(String,   FUpdates.Results.UrlButton); \    KEY(String,   FUpdates.Results.NewsUrl); \    KEYEX(Integer,FUpdates.Results.NewsSize.Width, L"NewsWidth"); \    KEYEX(Integer,FUpdates.Results.NewsSize.Height, L"NewsHeight"); \    KEY(String,   FUpdates.Results.DownloadUrl); \    KEY(Int64,    FUpdates.Results.DownloadSize); \    KEY(String,   FUpdates.Results.DownloadSha256); \    KEY(String,   FUpdates.Results.AuthenticationError); \    KEY(Bool,     FUpdates.Results.OpenGettingStarted); \    KEY(String,   FUpdates.Results.DownloadingUrl); \    KEYEX(Integer,FUpdates.Results.TipsSize.Width, L"TipsWidth"); \    KEYEX(Integer,FUpdates.Results.TipsSize.Height, L"TipsHeight"); \    KEY(String,   FUpdates.Results.TipsUrl); \    KEY(String,   FUpdates.Results.Tips); \    KEY(Integer,  FUpdates.Results.TipsIntervalDays); \    KEY(Integer,  FUpdates.Results.TipsIntervalRuns); \    KEY(String,   FUpdates.DotNetVersion); \    KEY(String,   FUpdates.ConsoleVersion); \  ); \  BLOCK(L"Interface\\Explorer", CANCREATE, \    KEYEX(String,  ScpExplorer.ToolbarsLayout, ToolbarsLayoutKey); \    KEY(String,  ScpExplorer.ToolbarsButtons); \    KEY(String,  ScpExplorer.DirViewParams); \    KEY(String,  ScpExplorer.LastLocalTargetDirectory); \    KEY(Bool,    ScpExplorer.SessionsTabs); \    KEY(Bool,    ScpExplorer.StatusBar); \    KEY(String,  ScpExplorer.WindowParams); \    KEY(Integer, ScpExplorer.ViewStyle); \    KEY(Bool,    ScpExplorer.ShowFullAddress); \    KEY(Bool,    ScpExplorer.DriveView); \    KEY(Integer, ScpExplorer.DriveViewWidth); \    KEY(Integer, ScpExplorer.DriveViewWidthPixelsPerInch); \  ); \  BLOCK(L"Interface\\Commander", CANCREATE, \    KEYEX(String,  ScpCommander.ToolbarsLayout, ToolbarsLayoutKey); \    KEY(String,  ScpCommander.ToolbarsButtons); \    KEY(Integer, ScpCommander.CurrentPanel); \    KEY(Float,   ScpCommander.LocalPanelWidth); \    KEY(Bool,    ScpCommander.SwappedPanels); \    KEY(Bool,    ScpCommander.SessionsTabs); \    KEY(Bool,    ScpCommander.StatusBar); \    KEY(String,  ScpCommander.WindowParams); \    KEYEX(Integer, ScpCommander.NortonLikeMode, L"ExplorerStyleSelection"); \    KEY(Bool,    ScpCommander.PreserveLocalDirectory); \    KEY(Bool,    ScpCommander.CompareByTime); \    KEY(Bool,    ScpCommander.CompareBySize); \    KEY(Bool,    ScpCommander.TreeOnLeft); \    KEY(Bool,    ScpCommander.ExplorerKeyboardShortcuts); \    KEY(Bool,    ScpCommander.SystemContextMenu); \  ); \  BLOCK(L"Interface\\Commander\\LocalPanel", CANCREATE, \    KEY(String,  ScpCommander.LocalPanel.DirViewParams); \    KEY(Bool,    ScpCommander.LocalPanel.StatusBar); \    KEY(Bool,    ScpCommander.LocalPanel.DriveView); \    KEY(Integer, ScpCommander.LocalPanel.DriveViewHeight); \    KEY(Integer, ScpCommander.LocalPanel.DriveViewHeightPixelsPerInch); \    KEY(Integer, ScpCommander.LocalPanel.DriveViewWidth); \    KEY(Integer, ScpCommander.LocalPanel.DriveViewWidthPixelsPerInch); \  ); \  BLOCK(L"Interface\\Commander\\RemotePanel", CANCREATE, \    KEY(String,  ScpCommander.RemotePanel.DirViewParams); \    KEY(Bool,    ScpCommander.RemotePanel.StatusBar); \    KEY(Bool,    ScpCommander.RemotePanel.DriveView); \    KEY(Integer, ScpCommander.RemotePanel.DriveViewHeight); \    KEY(Integer, ScpCommander.RemotePanel.DriveViewHeightPixelsPerInch); \    KEY(Integer, ScpCommander.RemotePanel.DriveViewWidth); \    KEY(Integer, ScpCommander.RemotePanel.DriveViewWidthPixelsPerInch); \  ); \  BLOCK(L"Security", CANCREATE, \    KEYEX(Bool,  FUseMasterPassword, L"UseMasterPassword"); \    KEYEX(String,FMasterPasswordVerifier, L"MasterPasswordVerifier"); \  );//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SaveData(THierarchicalStorage * Storage, bool All){  TCustomWinConfiguration::SaveData(Storage, All);  // duplicated from core\configuration.cpp  #define KEYEX(TYPE, VAR, NAME) Storage->Write ## TYPE(NAME, VAR)  REGCONFIG(true);  #undef KEYEX  if (Storage->OpenSubKey(L"Bookmarks", true))  {    FBookmarks->Save(Storage, All);    Storage->CloseSubKey();  }  if ((All && !FCustomCommandsDefaults) || FCustomCommandList->Modified)  {    FCustomCommandList->Save(Storage);  }  if ((All || FCustomCommandOptionsModified) &&      Storage->OpenSubKey(L"CustomCommandOptions", true))  {    Storage->ClearValues();    Storage->WriteValues(FCustomCommandOptions.get(), true);    Storage->CloseSubKey();    FCustomCommandOptionsModified = false;  }  if ((All || FEditorList->Modified) &&      Storage->OpenSubKey(L"Interface\\Editor", true, true))  try  {    FEditorList->Save(Storage);  }  __finally  {    Storage->CloseSubKey();  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::LoadFrom(THierarchicalStorage * Storage){  FLegacyEditor = new TEditorPreferences();  try  {    FLegacyEditor->LegacyDefaults();    TCustomWinConfiguration::LoadFrom(Storage);    // Following needs to be done even if there's no Configuration key in the storage,    // so it cannot be in LoadData    int EditorCount = FEditorList->Count;    if (EditorCount == 0)    {      TEditorPreferences * AlternativeEditor = NULL;      try      {        if (FLegacyEditor->Data->Editor == edInternal)        {          if (!FLegacyEditor->Data->ExternalEditor.IsEmpty())          {            AlternativeEditor = new TEditorPreferences(*FLegacyEditor);            AlternativeEditor->GetData()->Editor = edExternal;            FLegacyEditor->GetData()->ExternalEditor = L"";          }        }        else        {          if (FLegacyEditor->Data->ExternalEditor.IsEmpty())          {            FLegacyEditor->GetData()->Editor = edInternal;          }          else          {            AlternativeEditor = new TEditorPreferences(*FLegacyEditor);            AlternativeEditor->GetData()->Editor = edInternal;          }        }      }      catch(...)      {        delete AlternativeEditor;        throw;      }      FEditorList->Add(FLegacyEditor);      FLegacyEditor = NULL;      if (AlternativeEditor != NULL)      {        FEditorList->Add(AlternativeEditor);      }    }    // Additionally, this needs to be after Locale is loaded    LoadExtensionTranslations();    // and this after the ExtensionsDeleted, ExtensionsOrder and ExtensionsShortCuts are loaded    LoadExtensionList();  }  __finally  {    delete FLegacyEditor;    FLegacyEditor = NULL;  }  if (FUpdates.ConnectionType == -1)  {    FUpdates.ConnectionType = (FUpdates.ProxyHost.IsEmpty() ? ctAuto : ctProxy);  }  AddVersionToHistory();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::DoLoadExtensionList(  const UnicodeString & Path, const UnicodeString & PathId, TStringList * DeletedExtensions){  TSearchRecOwned SearchRec;  int FindAttrs = faReadOnly | faArchive;  if (FindFirstUnchecked(IncludeTrailingBackslash(Path) + L"*.*", FindAttrs, SearchRec) == 0)  {    do    {      UnicodeString Id = TCustomCommandType::GetExtensionId(SearchRec.Name);      if (!Id.IsEmpty())      {        Id = IncludeTrailingBackslash(PathId) + Id;        if (DeletedExtensions->IndexOf(Id) >= 0)        {          // reconstruct the list, so that we remove the commands that no longer exists          AddToList(FExtensionsDeleted, Id, L"|");        }        else        {          std::unique_ptr<TCustomCommandType> CustomCommand(new TCustomCommandType());          CustomCommand->Id = Id;          try          {            CustomCommand->LoadExtension(IncludeTrailingBackslash(Path) + SearchRec.Name);            FExtensionList->Add(CustomCommand.release());          }          catch (...)          {            // skip invalid extension files          }        }      }    }    while (FindNextChecked(SearchRec) == 0);  }}//---------------------------------------------------------------------------void __fastcall ParseExtensionList(TStrings * Strings, UnicodeString S){  while (!S.IsEmpty())  {    UnicodeString Extension = CutToChar(S, L'|', false);    Strings->Add(Extension);  }}//---------------------------------------------------------------------------const UnicodeString ExtensionsSubFolder(L"Extensions");const UnicodeString ExtensionsCommonPathId(L"common");const UnicodeString ExtensionsCommonExtPathId(L"commonext");const UnicodeString ExtensionsUserExtPathId(L"userext");//---------------------------------------------------------------------------static UnicodeString __fastcall ExtractExtensionBaseName(const UnicodeString & FileName){  UnicodeString S = ExtractFileName(FileName);  // ExtractFileNameOnly trims the last extension only, we want to trim all extensions  UnicodeString Result = CutToChar(S, L'.', true);  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::GetProvisionaryExtensionId(const UnicodeString & FileName){  return IncludeTrailingBackslash(ExtensionsUserExtPathId) + ExtractExtensionBaseName(FileName);}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::GetUserExtensionsPath(){  return IncludeTrailingBackslash(GetShellFolderPath(CSIDL_APPDATA)) + L"WinSCP\\" + ExtensionsSubFolder;}//---------------------------------------------------------------------------TStrings * __fastcall TWinConfiguration::GetExtensionsPaths(){  std::unique_ptr<TStrings> Result(new TStringList());  UnicodeString ExeParentPath = ExcludeTrailingBackslash(ExtractFilePath(Application->ExeName));  Result->Values[ExtensionsCommonPathId] = ExeParentPath;  UnicodeString CommonExtensions = IncludeTrailingBackslash(ExeParentPath) + ExtensionsSubFolder;  Result->Values[ExtensionsCommonExtPathId] = CommonExtensions;  Result->Values[ExtensionsUserExtPathId] = GetUserExtensionsPath();  return Result.release();}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::GetExtensionId(const UnicodeString & ExtensionPath){  UnicodeString Path = ExcludeTrailingBackslash(ExtractFilePath(ExtensionPath));  UnicodeString NameId = TCustomCommandType::GetExtensionId(ExtractFileName(ExtensionPath));  if (!NameId.IsEmpty())  {    std::unique_ptr<TStrings> ExtensionsPaths(GetExtensionsPaths());    for (int Index = 0; Index < ExtensionsPaths->Count; Index++)    {      if (IsPathToSameFile(Path, ExtensionsPaths->ValueFromIndex[Index]))      {        return IncludeTrailingBackslash(ExtensionsPaths->Names[Index]) + NameId;      }    }  }  return L"";}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::ReleaseExtensionTranslations(){  for (int Index = 0; Index < FExtensionTranslations->Count; Index++)  {    delete FExtensionTranslations->Objects[Index];  }  FExtensionTranslations->Clear();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::LoadExtensionTranslations(){  ReleaseExtensionTranslations();  FExtensionTranslations.reset(new TStringList());  int Index = EXTENSION_STRINGS;  UnicodeString S;  while (!((S = LoadStr(Index)).IsEmpty()))  {    UnicodeString ExtensionName = CutToChar(S, L'.', false);    UnicodeString Original = CutToChar(S, L'=', false);    UnicodeString Translation = S;    int ExtensionIndex = FExtensionTranslations->IndexOf(ExtensionName);    if (ExtensionIndex < 0)    {      ExtensionIndex = FExtensionTranslations->AddObject(ExtensionName, new TStringList());    }    TStringList * ExtensionTranslation = DebugNotNull(dynamic_cast<TStringList *>(FExtensionTranslations->Objects[ExtensionIndex]));    ExtensionTranslation->Values[Original] = Translation;    Index++;  }}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::UniqueExtensionName(const UnicodeString & ExtensionName, int Counter){  // See how the digits are removed in the ExtensionStringTranslation  return ExtensionName + IntToStr(Counter);}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::ExtensionStringTranslation(const UnicodeString & ExtensionId, const UnicodeString & S){  UnicodeString Result = S;  if (!ExtensionId.IsEmpty())  {    UnicodeString ExtensionName = ExtractFileName(ExtensionId);    int ExtensionIndex;    bool Retry;    do    {      ExtensionIndex = FExtensionTranslations->IndexOf(ExtensionName);      // Try without trailing digits, possibly added by UniqueExtensionName      Retry = (ExtensionIndex < 0) && !ExtensionName.IsEmpty() && IsDigit(*ExtensionName.LastChar());      if (Retry)      {        ExtensionName.SetLength(ExtensionName.Length() - 1);      }    }    while (Retry);    if (ExtensionIndex >= 0)    {      TStrings * ExtensionTranslation = DebugNotNull(dynamic_cast<TStrings *>(FExtensionTranslations->Objects[ExtensionIndex]));      int StringIndex = ExtensionTranslation->IndexOfName(S);      if (StringIndex >= 0)      {        Result = ExtensionTranslation->ValueFromIndex[StringIndex];      }    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::LoadExtensionList(){  FExtensionList->Clear();  std::unique_ptr<TStringList> DeletedExtensions(CreateSortedStringList());  ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted);  // will reconstruct the list in DoLoadExtensionList  FExtensionsDeleted = L"";  std::unique_ptr<TStrings> ExtensionsPaths(GetExtensionsPaths());  for (int Index = 0; Index < ExtensionsPaths->Count; Index++)  {    DoLoadExtensionList(ExtensionsPaths->ValueFromIndex[Index], ExtensionsPaths->Names[Index], DeletedExtensions.get());  }  std::unique_ptr<TStringList> OrderedExtensions(new TStringList());  ParseExtensionList(OrderedExtensions.get(), FExtensionsOrder);  FExtensionList->SortBy(OrderedExtensions.get());  UnicodeString ShortCuts = FExtensionsShortCuts;  while (!ShortCuts.IsEmpty())  {    UnicodeString S = CutToChar(ShortCuts, L'|', false);    TShortCut ShortCut = static_cast<TShortCut>(StrToInt(CutToChar(S, L'=', false)));    for (int Index = 0; Index < FExtensionList->Count; Index++)    {      if (FExtensionList->Commands[Index]->Id == S)      {        const_cast<TCustomCommandType *>(FExtensionList->Commands[Index])->ShortCut = ShortCut;      }    }  }}//---------------------------------------------------------------------------static UnicodeString KeyName(THierarchicalStorage * Storage, const UnicodeString & Name){  UnicodeString Result = Name;  if ((Result == ToolbarsLayoutKey) && !Storage->KeyExists(Result) && Storage->KeyExists(ToolbarsLayoutOldKey))  {    Result = ToolbarsLayoutOldKey;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::LoadData(THierarchicalStorage * Storage){  TCustomWinConfiguration::LoadData(Storage);  // duplicated from core\configuration.cpp  #define KEYEX(TYPE, VAR, NAME) VAR = Storage->Read ## TYPE(KeyName(Storage, NAME), VAR)  #pragma warn -eas  REGCONFIG(false);  #pragma warn +eas  #undef KEYEX  // to reflect changes to PanelFont  UpdateIconFont();  if (Storage->OpenSubKey(L"Bookmarks", false))  {    FBookmarks->Load(Storage);    Storage->CloseSubKey();  }  if (Storage->HasSubKey(L"CustomCommands"))  {    FCustomCommandList->Load(Storage);    FCustomCommandsDefaults = false;  }  else if (FCustomCommandList->Modified)  {    // can this (=reloading of configuration) even happen?    // if it does, shouldn't we reset default commands?    DebugFail();    FCustomCommandList->Clear();    FCustomCommandsDefaults = false;  }  FCustomCommandList->Reset();  if (Storage->OpenSubKey(L"CustomCommandOptions", false))  {    Storage->ReadValues(FCustomCommandOptions.get(), true);    Storage->CloseSubKey();    FCustomCommandOptionsModified = false;  }  if (Storage->OpenSubKey(L"Interface\\Editor", false, true))  try  {    FEditorList->Clear();    FEditorList->Load(Storage);  }  __finally  {    Storage->CloseSubKey();  }  // load legacy editor configuration  DebugAssert(FLegacyEditor != NULL);  if (Storage->OpenSubKey(L"Interface\\Editor", false, true))  {    try    {      FLegacyEditor->Load(Storage, true);    }    __finally    {      Storage->CloseSubKey();    }  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::LoadAdmin(THierarchicalStorage * Storage){  TCustomWinConfiguration::LoadAdmin(Storage);  FDisableOpenEdit = Storage->ReadBool(L"DisableOpenEdit", FDisableOpenEdit);  FDefaultUpdatesPeriod = Storage->ReadInteger(L"DefaultUpdatesPeriod", FDefaultUpdatesPeriod);  FMachineInstallations = Storage->ReadInteger(L"Installations", 0);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::CopyData(THierarchicalStorage * Source, THierarchicalStorage * Target){  TCustomWinConfiguration::CopyData(Source, Target);  if (Source->OpenSubKey(ConfigurationSubKey, false))  {    if (Target->OpenSubKey(ConfigurationSubKey, true))    {      Target->WriteString(L"JumpList", Source->ReadString(L"JumpList", L""));      Target->WriteString(L"JumpListWorkspaces", Source->ReadString(L"JumpListWorkspaces", L""));      Target->CloseSubKey();    }    Source->CloseSubKey();  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::ClearTemporaryLoginData(){  if (!FTemporaryKeyFile.IsEmpty())  {    DeleteFile(ApiPath(FTemporaryKeyFile));    FTemporaryKeyFile = L"";  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::AddVersionToHistory(){  int CurrentVersion = CompoundVersion;  int From = 1;  bool CurrentVersionPresent = false;  while (!CurrentVersionPresent && (From < FVersionHistory.Length()))  {    UnicodeString VersionInfo = CopyToChars(FVersionHistory, From, L";", true);    UnicodeString VersionStr = CutToChar(VersionInfo, L',', true);    int Version;    if (TryStrToInt(VersionStr, Version) &&        (Version == CurrentVersion))    {      CurrentVersionPresent = true;    }  }  if (!CurrentVersionPresent)  {    UnicodeString CurrentVersionInfo =      IntToStr(CurrentVersion) + L"," + GetReleaseType();    AddToList(FVersionHistory, CurrentVersionInfo, L';');  }  Usage->Set(L"AnyBetaUsed", AnyBetaInVersionHistory);}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::DoIsBeta(const UnicodeString & ReleaseType){  // What about "Development" release type?  return SameText(ReleaseType, L"beta") || SameText(ReleaseType, L"rc");}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetIsBeta(){  return DoIsBeta(GetReleaseType());}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetAnyBetaInVersionHistory(){  int From = 1;  bool AnyBeta = false;  while (!AnyBeta && (From < VersionHistory.Length()))  {    UnicodeString VersionInfo = CopyToChars(VersionHistory, From, L";", true);    CutToChar(VersionInfo, L',', true);    UnicodeString ReleaseType = CutToChar(VersionInfo, ',', true);    if (DoIsBeta(ReleaseType))    {      AnyBeta = true;    }  }  return AnyBeta;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetDDExtInstalled(){  if (FDDExtInstalled < 0)  {    if (IsWin64())    {      // WORKAROUND      // We cannot load 64-bit COM class into 32-bit process,      // so we fallback to querying registration keys      #define CLSID_SIZE 39      wchar_t ClassID[CLSID_SIZE];      StringFromGUID2(CLSID_ShellExtension, ClassID, LENOF(ClassID));      NULL_TERMINATE(ClassID);      UnicodeString SubKey = UnicodeString(L"CLSID\\") + ClassID;      HKEY HKey;      LONG Result = RegOpenKeyEx(HKEY_CLASSES_ROOT, SubKey.c_str(), 0, KEY_READ | KEY_WOW64_64KEY, &HKey);      if (Result == ERROR_SUCCESS)      {        RegCloseKey(HKey);        FDDExtInstalled = 1;      }      else      {        FDDExtInstalled = 0;      }    }    else    {      void * DragExtRef;      int CreateResult =        CoCreateInstance(CLSID_ShellExtension, NULL,          CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_IUnknown,          &DragExtRef);      bool Result = (CreateResult == S_OK);      FDDExtInstalled = (Result ? 1 : 0);      if (Result)      {        reinterpret_cast<IUnknown *>(DragExtRef)->Release();        CoFreeUnusedLibraries();      }    }  }  return (FDDExtInstalled > 0);}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::IsDDExtRunning(){  bool Result;  if (!DDExtInstalled)  {    Result = false;  }  else  {    HANDLE H = OpenMutex(SYNCHRONIZE, False, DRAG_EXT_RUNNING_MUTEX);    Result = (H != NULL);    CloseHandle(H);  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::IsDDExtBroken(){  return (GetWindowsBuild() >= 17134);}//---------------------------------------------------------------------------RawByteString __fastcall TWinConfiguration::StronglyRecryptPassword(RawByteString Password, UnicodeString Key){  RawByteString Dummy;  RawByteString Result;  if (GetExternalEncryptedPassword(Password, Dummy) ||      !FUseMasterPassword)  {    // already-strongly encrypted    // or no master password set, so we cannot strongly-encrypt it    Result = Password;  }  else  {    UnicodeString PasswordText =      TCustomWinConfiguration::DecryptPassword(Password, Key);    if (!PasswordText.IsEmpty())    {      // can be not set for instance, when editing=>saving site with no prior password      AskForMasterPasswordIfNotSet();      Password = ScramblePassword(PasswordText);      AES256EncyptWithMAC(Password, FPlainMasterPasswordEncrypt, Result);      Result = SetExternalEncryptedPassword(Result);    }  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::DecryptPassword(RawByteString Password, UnicodeString Key){  UnicodeString Result;  RawByteString Buf;  if (GetExternalEncryptedPassword(Password, Buf))  {    if (FDontDecryptPasswords == 0)    {      // As opposite to AskForMasterPasswordIfNotSet, we test here      // for decrypt password. This is important while recrypting password,      // when clearing master password, when encrypt password is already empty.      if (FPlainMasterPasswordDecrypt.IsEmpty())      {        AskForMasterPassword();      }      if (!AES256DecryptWithMAC(Buf, FPlainMasterPasswordDecrypt, Buf) ||          !UnscramblePassword(Buf, Result))      {        throw Exception(LoadStr(DECRYPT_PASSWORD_ERROR));      }    }    else    {      Abort();    }  }  else  {    Result = TCustomWinConfiguration::DecryptPassword(Password, Key);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetMasterPassword(UnicodeString value){  // just stores the plain-text version of the password  // we can get here even if master password is off,  // when we encounter stray encrypted password (e.g. manually imported site),  // make sure we do not set encrypt password to avoid starting encrypting  // new passwords  // (this is bit of edge case, not really well tested)  if (!FUseMasterPassword)  {    FPlainMasterPasswordDecrypt = value;  }  else if (DebugAlwaysTrue(FUseMasterPassword) &&      DebugAlwaysTrue(ValidateMasterPassword(value)))  {    FPlainMasterPasswordEncrypt = value;    FPlainMasterPasswordDecrypt = value;  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::ChangeMasterPassword(  UnicodeString value, TStrings * RecryptPasswordErrors){  RawByteString Verifier;  AES256CreateVerifier(value, Verifier);  FMasterPasswordVerifier = BytesToHex(Verifier);  FPlainMasterPasswordEncrypt = value;  FUseMasterPassword = true;  try  {    RecryptPasswords(RecryptPasswordErrors);  }  __finally  {    FPlainMasterPasswordDecrypt = value;  }}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::ValidateMasterPassword(UnicodeString value){  DebugAssert(UseMasterPassword);  DebugAssert(!FMasterPasswordVerifier.IsEmpty());  bool Result = AES256Verify(value, HexToBytes(FMasterPasswordVerifier));  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::ClearMasterPassword(TStrings * RecryptPasswordErrors){  FMasterPasswordVerifier = L"";  FUseMasterPassword = false;  Shred(FPlainMasterPasswordEncrypt);  try  {    RecryptPasswords(RecryptPasswordErrors);  }  __finally  {    Shred(FPlainMasterPasswordDecrypt);  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::AskForMasterPassword(){  if (FMasterPasswordSession > 0)  {    if (FMasterPasswordSessionAsked)    {      Abort();    }    // set before call to OnMasterPasswordPrompt as it may abort    FMasterPasswordSessionAsked = true;  }  if (FOnMasterPasswordPrompt == NULL)  {    throw Exception(L"Master password handler not set");  }  else  {    FOnMasterPasswordPrompt();    DebugAssert(!FPlainMasterPasswordDecrypt.IsEmpty());  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::AskForMasterPasswordIfNotSet(){  if (FPlainMasterPasswordEncrypt.IsEmpty())  {    AskForMasterPassword();  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::BeginMasterPasswordSession(){  // We do not expect nesting  DebugAssert(FMasterPasswordSession == 0);  DebugAssert(!FMasterPasswordSessionAsked || (FMasterPasswordSession > 0));  // This should better be thread-specific  FMasterPasswordSession++;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::EndMasterPasswordSession(){  if (DebugAlwaysTrue(FMasterPasswordSession > 0))  {    FMasterPasswordSession--;  }  FMasterPasswordSessionAsked = false;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDTransferConfirmation(TAutoSwitch value){  SET_CONFIG_PROPERTY(DDTransferConfirmation);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDTemporaryDirectory(UnicodeString value){  SET_CONFIG_PROPERTY(DDTemporaryDirectory);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDDrives(UnicodeString value){  SET_CONFIG_PROPERTY(DDDrives);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDFakeFile(bool value){  SET_CONFIG_PROPERTY(DDFakeFile);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDExtTimeout(int value){  SET_CONFIG_PROPERTY(DDExtTimeout);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDWarnLackOfTempSpace(bool value){  SET_CONFIG_PROPERTY(DDWarnLackOfTempSpace);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDDWarnLackOfTempSpaceRatio(double value){  SET_CONFIG_PROPERTY(DDWarnLackOfTempSpaceRatio);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetScpExplorer(TScpExplorerConfiguration value){  SET_CONFIG_PROPERTY(ScpExplorer);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetScpCommander(TScpCommanderConfiguration value){  SET_CONFIG_PROPERTY(ScpCommander);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetEditor(TEditorConfiguration value){  SET_CONFIG_PROPERTY(Editor);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetQueueView(TQueueViewConfiguration value){  SET_CONFIG_PROPERTY(QueueView);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetEnableQueueByDefault(bool value){  SET_CONFIG_PROPERTY(EnableQueueByDefault);}//---------------------------------------------------------------------------TUpdatesConfiguration __fastcall TWinConfiguration::GetUpdates(){  TUpdatesConfiguration Result;  {    TGuard Guard(FCriticalSection);    Result = FUpdates;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetUpdates(TUpdatesConfiguration value){  TGuard Guard(FCriticalSection);  // do not use SET_CONFIG_PROPERTY to avoid OnChange handler call (not synchronized)  FUpdates = value;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetVersionHistory(UnicodeString value){  SET_CONFIG_PROPERTY(VersionHistory);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDeleteToRecycleBin(bool value){  SET_CONFIG_PROPERTY(DeleteToRecycleBin);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetSelectDirectories(bool value){  SET_CONFIG_PROPERTY(SelectDirectories);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetShowHiddenFiles(bool value){  SET_CONFIG_PROPERTY(ShowHiddenFiles);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetFormatSizeBytes(TFormatBytesStyle value){  SET_CONFIG_PROPERTY(FormatSizeBytes);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetShowInaccesibleDirectories(bool value){  SET_CONFIG_PROPERTY(ShowInaccesibleDirectories);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetConfirmTransferring(bool value){  SET_CONFIG_PROPERTY(ConfirmTransferring);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetConfirmDeleting(bool value){  SET_CONFIG_PROPERTY(ConfirmDeleting);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetConfirmRecycling(bool value){  SET_CONFIG_PROPERTY(ConfirmRecycling);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetUseLocationProfiles(bool value){  SET_CONFIG_PROPERTY(UseLocationProfiles);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetUseSharedBookmarks(bool value){  SET_CONFIG_PROPERTY(UseSharedBookmarks);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetConfirmClosingSession(bool value){  SET_CONFIG_PROPERTY(ConfirmClosingSession);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDoubleClickAction(TDoubleClickAction value){  SET_CONFIG_PROPERTY(DoubleClickAction);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetCopyOnDoubleClickConfirmation(bool value){  SET_CONFIG_PROPERTY(CopyOnDoubleClickConfirmation);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDimmHiddenFiles(bool value){  SET_CONFIG_PROPERTY(DimmHiddenFiles);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetRenameWholeName(bool value){  SET_CONFIG_PROPERTY(RenameWholeName);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoStartSession(UnicodeString value){  SET_CONFIG_PROPERTY(AutoStartSession);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetExpertMode(bool value){  SET_CONFIG_PROPERTY(ExpertMode);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDefaultDirIsHome(bool value){  SET_CONFIG_PROPERTY(DefaultDirIsHome);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTemporaryDirectoryAppendSession(bool value){  SET_CONFIG_PROPERTY(TemporaryDirectoryAppendSession);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTemporaryDirectoryAppendPath(bool value){  SET_CONFIG_PROPERTY(TemporaryDirectoryAppendPath);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTemporaryDirectoryDeterministic(bool value){  SET_CONFIG_PROPERTY(TemporaryDirectoryDeterministic);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTemporaryDirectoryCleanup(bool value){  SET_CONFIG_PROPERTY(TemporaryDirectoryCleanup);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetConfirmTemporaryDirectoryCleanup(bool value){  SET_CONFIG_PROPERTY(ConfirmTemporaryDirectoryCleanup);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetPreservePanelState(bool value){  SET_CONFIG_PROPERTY(PreservePanelState);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetDarkTheme(TAutoSwitch value){  SET_CONFIG_PROPERTY_EX(DarkTheme, ConfigureInterface());}//---------------------------------------------------------------------------static int __fastcall SysDarkTheme(HKEY RootKey){  std::unique_ptr<TRegistry> Registry(new TRegistry());  Registry->RootKey = RootKey;  UnicodeString ThemesPersonalizeKey = L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize";  UnicodeString AppsUseLightThemeValue = L"AppsUseLightTheme";  int Result = -1;  if (Registry->OpenKeyReadOnly(ThemesPersonalizeKey) &&      Registry->ValueExists(AppsUseLightThemeValue))  {    Result = Registry->ReadBool(AppsUseLightThemeValue) ? 0 : 1;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::ResetSysDarkTheme(){  FSysDarkTheme = -1;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::UseDarkTheme(){  if (FSysDarkTheme < 0)  {    FSysDarkTheme = SysDarkTheme(HKEY_CURRENT_USER);    if (FSysDarkTheme < 0)    {      FSysDarkTheme = SysDarkTheme(HKEY_LOCAL_MACHINE);      if (FSysDarkTheme < 0)      {        FSysDarkTheme = 0;      }    }  }  switch (WinConfiguration->DarkTheme)  {    case asOn:      return true;    case asOff:      return false;    default:      return (FSysDarkTheme > 0);  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetLastStoredSession(UnicodeString value){  SET_CONFIG_PROPERTY(LastStoredSession);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoSaveWorkspace(bool value){  SET_CONFIG_PROPERTY(AutoSaveWorkspace);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoSaveWorkspacePasswords(bool value){  SET_CONFIG_PROPERTY(AutoSaveWorkspacePasswords);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoWorkspace(UnicodeString value){  SET_CONFIG_PROPERTY(AutoWorkspace);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetPathInCaption(TPathInCaption value){  SET_CONFIG_PROPERTY(PathInCaption);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetMinimizeToTray(bool value){  SET_CONFIG_PROPERTY(MinimizeToTray);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::MinimizeToTrayOnce(){  FMinimizeToTrayOnce = true;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetMinimizeToTray(){  bool Result = FMinimizeToTrayOnce || FMinimizeToTray;  FMinimizeToTrayOnce = false;  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetBalloonNotifications(bool value){  SET_CONFIG_PROPERTY(BalloonNotifications);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetNotificationsTimeout(unsigned int value){  SET_CONFIG_PROPERTY(NotificationsTimeout);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetNotificationsStickTime(unsigned int value){  SET_CONFIG_PROPERTY(NotificationsStickTime);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetCopyParamAutoSelectNotice(bool value){  SET_CONFIG_PROPERTY(CopyParamAutoSelectNotice);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetLockToolbars(bool value){  SET_CONFIG_PROPERTY(LockToolbars);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetSelectiveToolbarText(bool value){  SET_CONFIG_PROPERTY(SelectiveToolbarText);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoOpenInPutty(bool value){  SET_CONFIG_PROPERTY(AutoOpenInPutty);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetRefreshRemotePanel(bool value){  SET_CONFIG_PROPERTY(RefreshRemotePanel);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetRefreshRemotePanelInterval(TDateTime value){  SET_CONFIG_PROPERTY(RefreshRemotePanelInterval);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::UpdateIconFont(){  UpdateDesktopFont();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetPanelFont(const TFontConfiguration & value){  SET_CONFIG_PROPERTY_EX(PanelFont, UpdateIconFont());}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetNaturalOrderNumericalSorting(bool value){  SET_CONFIG_PROPERTY(NaturalOrderNumericalSorting);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetFullRowSelect(bool value){  SET_CONFIG_PROPERTY(FullRowSelect);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetOfferedEditorAutoConfig(bool value){  SET_CONFIG_PROPERTY(OfferedEditorAutoConfig);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetOpenedStoredSessionFolders(UnicodeString value){  SET_CONFIG_PROPERTY(OpenedStoredSessionFolders);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetAutoImportedFromPuttyOrFilezilla(bool value){  SET_CONFIG_PROPERTY(AutoImportedFromPuttyOrFilezilla);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetGenerateUrlComponents(int value){  SET_CONFIG_PROPERTY(GenerateUrlComponents);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetGenerateUrlCodeTarget(TGenerateUrlCodeTarget value){  SET_CONFIG_PROPERTY(GenerateUrlCodeTarget);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetGenerateUrlScriptFormat(TScriptFormat value){  SET_CONFIG_PROPERTY(GenerateUrlScriptFormat);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetGenerateUrlAssemblyLanguage(TAssemblyLanguage value){  SET_CONFIG_PROPERTY(GenerateUrlAssemblyLanguage);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetExternalSessionInExistingInstance(bool value){  SET_CONFIG_PROPERTY(ExternalSessionInExistingInstance);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetKeepOpenWhenNoSession(bool value){  SET_CONFIG_PROPERTY(KeepOpenWhenNoSession);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetLocalIconsByExt(bool value){  SET_CONFIG_PROPERTY(LocalIconsByExt);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetBidiModeOverride(TLocaleFlagOverride value){  SET_CONFIG_PROPERTY(BidiModeOverride);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetFlipChildrenOverride(TLocaleFlagOverride value){  SET_CONFIG_PROPERTY(FlipChildrenOverride);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetShowTips(bool value){  SET_CONFIG_PROPERTY(ShowTips);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTipsSeen(UnicodeString value){  SET_CONFIG_PROPERTY(TipsSeen);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTipsShown(TDateTime value){  SET_CONFIG_PROPERTY(TipsShown);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetFileColors(UnicodeString value){  SET_CONFIG_PROPERTY(FileColors);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetRunsSinceLastTip(int value){  SET_CONFIG_PROPERTY(RunsSinceLastTip);}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetHonorDrivePolicy(){  return DriveInfo->HonorDrivePolicy;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetHonorDrivePolicy(bool value){  if (HonorDrivePolicy != value)  {    DriveInfo->HonorDrivePolicy = value;    Changed();  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetCustomCommandList(TCustomCommandList * value){  DebugAssert(FCustomCommandList);  if (!FCustomCommandList->Equals(value))  {    FCustomCommandList->Assign(value);    FCustomCommandsDefaults = false;  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetExtensionList(TCustomCommandList * value){  if (!ExtensionList->Equals(value))  {    std::unique_ptr<TStringList> DeletedExtensions(CreateSortedStringList());    ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted);    for (int Index = 0; Index < ExtensionList->Count; Index++)    {      const TCustomCommandType * CustomCommand = ExtensionList->Commands[Index];      int Index = value->FindIndexByFileName(CustomCommand->FileName);      if (Index < 0)      {        if (FileExists(CustomCommand->FileName) &&            !DeleteFile(CustomCommand->FileName))        {          DeletedExtensions->Add(CustomCommand->Id);        }      }    }    FExtensionsOrder = L"";    for (int Index = 0; Index < value->Count; Index++)    {      const TCustomCommandType * CustomCommand = value->Commands[Index];      AddToList(FExtensionsOrder, CustomCommand->Id, L"|");      int DeletedIndex = DeletedExtensions->IndexOf(CustomCommand->Id);      if (DeletedIndex >= 0)      {        DeletedExtensions->Delete(DeletedIndex);      }    }    FExtensionList->Assign(value);    FExtensionsDeleted = L"";    for (int Index = 0; Index < DeletedExtensions->Count; Index++)    {      AddToList(FExtensionsDeleted, DeletedExtensions->Strings[Index], L"|");    }    FExtensionsShortCuts = L"";    for (int Index = 0; Index < value->Count; Index++)    {      const TCustomCommandType * Extension = value->Commands[Index];      if (Extension->HasCustomShortCut())      {        AddToList(FExtensionsShortCuts, FORMAT(L"%d=%s", (Extension->ShortCut, Extension->Id)), L"|");      }    }    Changed();  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::CustomCommandShortCuts(TShortCuts & ShortCuts) const{  CustomCommandList->ShortCuts(ShortCuts);  ExtensionList->ShortCuts(ShortCuts);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetBookmarks(UnicodeString Key,  TBookmarkList * value){  FBookmarks->Bookmarks[Key] = value;  Changed();}//---------------------------------------------------------------------------TBookmarkList * __fastcall TWinConfiguration::GetBookmarks(UnicodeString Key){  return FBookmarks->Bookmarks[Key];}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetSharedBookmarks(TBookmarkList * value){  FBookmarks->SharedBookmarks = value;  Changed();}//---------------------------------------------------------------------------TBookmarkList * __fastcall TWinConfiguration::GetSharedBookmarks(){  return FBookmarks->SharedBookmarks;}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::GetDefaultKeyFile(){  return (!FDefaultKeyFile.IsEmpty() ? FDefaultKeyFile : FTemporaryKeyFile);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetLastMonitor(int value){  ::SetLastMonitor(value);}//---------------------------------------------------------------------------int __fastcall TWinConfiguration::GetLastMonitor(){  return ::GetLastMonitor();}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::ExpandedTemporaryDirectory(){  UnicodeString Result =    ExpandFileName(ExpandEnvironmentVariables(DDTemporaryDirectory));  if (Result.IsEmpty())  {    Result = SystemTemporaryDirectory();  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TWinConfiguration::TemporaryDir(bool Mask){  return UniqTempDir(ExpandedTemporaryDirectory(), L"scp", Mask);}//---------------------------------------------------------------------------TStrings * __fastcall TWinConfiguration::FindTemporaryFolders(){  TStrings * Result = new TStringList();  try  {    TSearchRecOwned SRec;    UnicodeString Mask = TemporaryDir(true);    UnicodeString Directory = ExtractFilePath(Mask);    if (FindFirstUnchecked(Mask, faDirectory | faHidden, SRec) == 0)    {      do      {        if (SRec.IsDirectory())        {          Result->Add(Directory + SRec.Name);        }      }      while (FindNextChecked(SRec) == 0);    }    if (Result->Count == 0)    {      delete Result;      Result = NULL;    }  }  catch(...)  {    delete Result;    throw;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::CleanupTemporaryFolders(TStrings * Folders){  UnicodeString ErrorList;  TStrings * F;  if (Folders == NULL)  {    F = FindTemporaryFolders();  }  else  {    F = Folders;  }  if (F != NULL)  {    try    {      if (DebugAlwaysTrue(F->Count > 0))      {        Usage->Inc(L"TemporaryDirectoryCleanups");      }      for (int i = 0; i < F->Count; i++)      {        if (!DeleteDirectory(F->Strings[i]))        {          if (!ErrorList.IsEmpty())          {            ErrorList += L"\n";          }          ErrorList += F->Strings[i];        }      }    }    __finally    {      if (Folders == NULL)      {        delete F;      }    }    if (!ErrorList.IsEmpty())    {      throw ExtException(LoadStr(CLEANUP_TEMP_ERROR), ErrorList);    }  }}//---------------------------------------------------------------------------int __fastcall TWinConfiguration::GetResourceModuleCompleteness(HINSTANCE Module){  UnicodeString CompletenessStr = LoadStrFrom(Module, TRANSLATION_COMPLETENESS);  return StrToIntDef(CompletenessStr, -1);}//---------------------------------------------------------------------------int __fastcall TWinConfiguration::GetLocaleCompletenessTreshold(){  return 80;}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::IsTranslationComplete(HINSTANCE Module){  return (GetResourceModuleCompleteness(Module) >= LocaleCompletenessTreshold);}//---------------------------------------------------------------------------HINSTANCE __fastcall TWinConfiguration::LoadNewResourceModule(LCID ALocale,  UnicodeString & FileName){  HINSTANCE Instance = TCustomWinConfiguration::LoadNewResourceModule(ALocale, FileName);  if (Instance != NULL)  {    try    {      CheckTranslationVersion(FileName, false);    }    catch(...)    {      FreeResourceModule(Instance);      throw;    }  }  return Instance;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::CheckTranslationVersion(const UnicodeString FileName,  bool InternalLocaleOnError){  UnicodeString TranslationProductVersion = GetFileProductVersion(FileName);  UnicodeString TranslationProductName = GetFileProductName(FileName);  if ((ProductName != TranslationProductName) ||      (ProductVersion != TranslationProductVersion))  {    if (InternalLocaleOnError)    {      LocaleSafe = InternalLocale();    }    if (TranslationProductName.IsEmpty() || TranslationProductVersion.IsEmpty())    {      throw Exception(FMTLOAD(UNKNOWN_TRANSLATION, (FileName)));    }    else    {      throw Exception(FMTLOAD(INCOMPATIBLE_TRANSLATION,        (FileName, TranslationProductName, TranslationProductVersion)));    }  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::CheckDefaultTranslation(){  if (!FInvalidDefaultTranslationMessage.IsEmpty())  {    MoreMessageDialog(FMTLOAD(INVALID_DEFAULT_TRANSLATION,      (FInvalidDefaultTranslationMessage)), NULL, qtWarning, qaOK, HELP_NONE);  }}//---------------------------------------------------------------------------const TEditorPreferences * __fastcall TWinConfiguration::DefaultEditorForFile(  const UnicodeString FileName, bool Local, const TFileMasks::TParams & MaskParams){  return FEditorList->Find(FileName, Local, MaskParams);}//---------------------------------------------------------------------------const TEditorList * __fastcall TWinConfiguration::GetEditorList(){  return FEditorList;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetEditorList(const TEditorList * value){  if (!(*FEditorList == *value))  {    *FEditorList = *value;    Changed();  }}//---------------------------------------------------------------------------TStrings * __fastcall TWinConfiguration::GetCustomCommandOptions(){  return FCustomCommandOptions.get();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetCustomCommandOptions(TStrings * value){  if (!FCustomCommandOptions->Equals(value))  {    FCustomCommandOptions->Assign(value);    FCustomCommandOptionsModified = true;  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetLockedInterface(bool value){  SET_CONFIG_PROPERTY(LockedInterface);}//---------------------------------------------------------------------------bool __fastcall TWinConfiguration::GetTimeoutShellOperations(){  return ::TimeoutShellOperations;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTimeoutShellOperations(bool value){  ::TimeoutShellOperations = value;}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SetTimeoutShellIconRetrieval(bool value){  SET_CONFIG_PROPERTY(TimeoutShellIconRetrieval);}//---------------------------------------------------------------------------TStringList * __fastcall TWinConfiguration::LoadJumpList(  THierarchicalStorage * Storage, UnicodeString Name){  UnicodeString JumpList = Storage->ReadString(Name, L"");  std::unique_ptr<TStringList> List(new TStringList());  List->CaseSensitive = false;  List->CommaText = JumpList;  return List.release();}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::SaveJumpList(  THierarchicalStorage * Storage, UnicodeString Name, TStringList * List){  UnicodeString JumpList = Storage->ReadString(Name, L"");  UnicodeString NewJumpList = List->CommaText;  if (NewJumpList != JumpList)  {    Storage->WriteString(Name, NewJumpList);  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::TrimJumpList(TStringList * List){  while (List->Count > 30)  {    List->Delete(List->Count - 1);  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::UpdateEntryInJumpList(  bool Session, const UnicodeString & Name, bool Add){  THierarchicalStorage * Storage = CreateConfigStorage();  try  {    FDontDecryptPasswords++;    Storage->AccessMode = smReadWrite;    if (Storage->OpenSubKey(ConfigurationSubKey, true))    {      std::unique_ptr<TStringList> ListSessions(LoadJumpList(Storage, L"JumpList"));      std::unique_ptr<TStringList> ListWorkspaces(LoadJumpList(Storage, L"JumpListWorkspaces"));      if (!Name.IsEmpty())      {        TStringList * List = (Session ? ListSessions : ListWorkspaces).get();        int Index = List->IndexOf(Name);        if (Index >= 0)        {          List->Delete(Index);        }        if (Add)        {          List->Insert(0, Name);        }      }      TrimJumpList(ListSessions.get());      TrimJumpList(ListWorkspaces.get());      ::UpdateJumpList(ListSessions.get(), ListWorkspaces.get());      SaveJumpList(Storage, L"JumpList", ListSessions.get());      SaveJumpList(Storage, L"JumpListWorkspaces", ListWorkspaces.get());    }  }  __finally  {    FDontDecryptPasswords--;    delete Storage;  }}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::AddSessionToJumpList(UnicodeString SessionName){  UpdateEntryInJumpList(true, SessionName, true);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::DeleteSessionFromJumpList(UnicodeString SessionName){  UpdateEntryInJumpList(true, SessionName, false);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::AddWorkspaceToJumpList(UnicodeString Workspace){  UpdateEntryInJumpList(false, Workspace, true);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::DeleteWorkspaceFromJumpList(UnicodeString Workspace){  UpdateEntryInJumpList(false, Workspace, false);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::UpdateJumpList(){  AddSessionToJumpList(L"");}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::UpdateStaticUsage(){  // This is here because of TStoredSessionList::UpdateStaticUsage() call  // from TConfiguration::UpdateStaticUsage()  TAutoNestingCounter DontDecryptPasswordsAutoCounter(FDontDecryptPasswords);  TCustomWinConfiguration::UpdateStaticUsage();  Usage->Set(L"Beta", IsBeta);  Usage->Set(L"Interface", Interface);  Usage->Set(L"ThemeDark", UseDarkTheme());  Usage->Set(L"CustomCommandsCount", (FCustomCommandsDefaults ? 0 : FCustomCommandList->Count));  Usage->Set(L"UsingLocationProfiles", UseLocationProfiles);  Usage->Set(L"UsingMasterPassword", UseMasterPassword);  Usage->Set(L"UsingAutoSaveWorkspace", AutoSaveWorkspace);  Usage->Set(L"TreeVisible",    (Interface == ifExplorer) ?      ScpExplorer.DriveView :      (ScpCommander.LocalPanel.DriveView || ScpCommander.RemotePanel.DriveView));  Usage->Set(L"MinimizeToTray", MinimizeToTray);  UnicodeString ToolbarsButtons = (Interface == ifExplorer) ? ScpExplorer.ToolbarsButtons : ScpCommander.ToolbarsButtons;  Usage->Set(L"AnyHiddenToolbarButtons", !ToolbarsButtons.IsEmpty());  Usage->Set(L"FileColors", !FileColors.IsEmpty());  Usage->Set(L"DragDropDrives", !DDDrives.IsEmpty());  Usage->Set(L"ShowingTips", ShowTips);  TipsUpdateStaticUsage();  Usage->Set(L"CommanderNortonLikeMode", int(ScpCommander.NortonLikeMode));  Usage->Set(L"CommanderExplorerKeyboardShortcuts", ScpCommander.ExplorerKeyboardShortcuts);  Usage->Set(L"ExplorerViewStyle", ScpExplorer.ViewStyle);  Usage->Set(L"LastMonitor", LastMonitor);  UnicodeString ExternalEditors;  for (int Index = 0; Index < EditorList->Count; Index++)  {    const TEditorPreferences * Editor = EditorList->Editors[Index];    if (Editor->Data->Editor == edExternal)    {      AddToList(ExternalEditors, Editor->ExtractExternalEditorName(), ",");    }  }  Usage->Set(L"ExternalEditors", ExternalEditors);  if (LastMachineInstallations < FMachineInstallations)  {    Usage->Inc(L"InstallationsMachine", (FMachineInstallations - LastMachineInstallations));    LastMachineInstallations = FMachineInstallations;  }  int ExtensionsPortable = 0;  int ExtensionsInstalled = 0;  int ExtensionsUser = 0;  for (int Index = 0; Index < FExtensionList->Count; Index++)  {    const TCustomCommandType * CustomCommand = FExtensionList->Commands[Index];    UnicodeString PathId = ExcludeTrailingBackslash(ExtractFilePath(CustomCommand->Id));    if (SameText(PathId, ExtensionsCommonPathId))    {      ExtensionsPortable++;    }    else if (SameText(PathId, ExtensionsCommonExtPathId))    {      ExtensionsInstalled++;    }    else if (SameText(PathId, ExtensionsUserExtPathId))    {      ExtensionsUser++;    }  }  Usage->Set(L"ExtensionsPortableCount", (ExtensionsPortable));  Usage->Set(L"ExtensionsInstalledCount", (ExtensionsInstalled));  Usage->Set(L"ExtensionsUserCount", (ExtensionsUser));  std::unique_ptr<TStringList> DeletedExtensions(CreateSortedStringList());  ParseExtensionList(DeletedExtensions.get(), FExtensionsDeleted);  Usage->Set(L"ExtensionsDeleted", (DeletedExtensions->Count));}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::RestoreFont(  const TFontConfiguration & Configuration, TFont * Font){  Font->Name = Configuration.FontName;  Font->Size = Configuration.FontSize;  Font->Charset = Configuration.FontCharset;  Font->Style = IntToFontStyles(Configuration.FontStyle);}//---------------------------------------------------------------------------void __fastcall TWinConfiguration::StoreFont(  TFont * Font, TFontConfiguration & Configuration){  Configuration.FontName = Font->Name;  Configuration.FontSize = Font->Size;  Configuration.FontCharset = Font->Charset;  Configuration.FontStyle = FontStylesToInt(Font->Style);}//---------------------------------------------------------------------------//---------------------------------------------------------------------------__fastcall TCustomCommandType::TCustomCommandType() :  FParams(0), FShortCut(0), FShortCutOriginal(0){}//---------------------------------------------------------------------------__fastcall TCustomCommandType::TCustomCommandType(const TCustomCommandType & Other) :  FName(Other.FName),  FCommand(Other.FCommand),  FParams(Other.FParams),  FShortCut(Other.FShortCut),  FShortCutOriginal(Other.FShortCutOriginal),  FId(Other.FId),  FFileName(Other.FFileName),  FDescription(Other.FDescription),  FHomePage(Other.FHomePage),  FOptionsPage(Other.FOptionsPage),  FOptions(Other.FOptions){}//---------------------------------------------------------------------------TCustomCommandType & TCustomCommandType::operator=(const TCustomCommandType & Other){  FName = Other.FName;  FCommand = Other.FCommand;  FParams = Other.FParams;  FShortCut = Other.FShortCut;  FShortCutOriginal = Other.FShortCutOriginal;  FId = Other.FId;  FFileName = Other.FFileName;  FDescription = Other.FDescription;  FHomePage = Other.FHomePage;  FOptionsPage = Other.FOptionsPage;  FOptions = Other.FOptions;  return *this;}//---------------------------------------------------------------------------bool __fastcall TCustomCommandType::Equals(const TCustomCommandType * Other) const{  return    (FName == Other->FName) &&    (FCommand == Other->FCommand) &&    (FParams == Other->FParams) &&    (FShortCut == Other->FShortCut) &&    (FShortCutOriginal == Other->FShortCutOriginal) &&    (FId == Other->FId) &&    (FFileName == Other->FFileName) &&    (FDescription == Other->FDescription) &&    (FHomePage == Other->FHomePage) &&    (FOptionsPage == Other->FOptionsPage) &&    (FOptions == Other->FOptions);}//---------------------------------------------------------------------------const UnicodeString ExtensionNameDirective(L"name");const UnicodeString ExtensionCommandDirective(L"command");const wchar_t ExtensionMark = L'@';const UnicodeString WinSCPExtensionExt(L".WinSCPextension");//---------------------------------------------------------------------------UnicodeString __fastcall TCustomCommandType::GetExtensionId(const UnicodeString & Name){  UnicodeString Result;  int P = Pos(UpperCase(WinSCPExtensionExt), UpperCase(Name));  // Ends with Ext or there's another extension after it  if ((P > 1) &&      ((Name.Length() == (P + WinSCPExtensionExt.Length() - 1)) || (Name[P + WinSCPExtensionExt.Length()] == L'.')))  {    Result = Name.SubString(1, P - 1);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TCustomCommandType::LoadExtension(const UnicodeString & Path){  std::unique_ptr<TStringList> Lines(new TStringList());  FileName = Path;  LoadScriptFromFile(Path, Lines.get());  LoadExtension(Lines.get(), Path);  Command = ReplaceStr(Command, L"%EXTENSION_PATH%", Path);}//---------------------------------------------------------------------------void __fastcall TCustomCommandType::LoadExtension(TStrings * Lines, const UnicodeString & PathForBaseName){  Params = ccLocal;  bool AnythingFound = false;  std::set<UnicodeString> OptionIds;  UnicodeString ExtensionBaseName = ExtractExtensionBaseName(PathForBaseName);  UnicodeString ExtensionLine;  bool Break = false;  for (int Index = 0; !Break && (Index < Lines->Count); Index++)  {    UnicodeString Line = Lines->Strings[Index].Trim();    if (!Line.IsEmpty())    {      Line = ReplaceChar(Line, L'\t', L' ');      bool IsComment = false;      if (StartsText(ExtensionMark, Line))      {        IsComment = true;      }      else if (StartsText(L"rem ", Line))      {        IsComment = true;        Line.Delete(1, 4);      }      else if (StartsText(L"#", Line) || StartsText(L";", Line) || StartsText(L"'", Line))      {        IsComment = true;        Line.Delete(1, 1);      }      else if (StartsText(L"//", Line))      {        IsComment = true;        Line.Delete(1, 2);      }      if (!IsComment)      {        Break = true;        // ignore this and later lines, but finish processing the previous line with ^, if any        Line = L"";      }      else      {        Line = Line.Trim();      }    }    bool Continuation = (Line.Length() > 0) && (Line[Line.Length()] == L'^');    if (Continuation)    {      Line = Line.SubString(1, Line.Length() - 1).Trim();    }    AddToList(ExtensionLine, Line, L" ");    if (!Continuation)    {      int P;      if (!ExtensionLine.IsEmpty() && (ExtensionLine[1] == ExtensionMark) && ((P = Pos(L" ", ExtensionLine)) >= 2))      {        UnicodeString Key = ExtensionLine.SubString(2, P - 2).LowerCase();        UnicodeString Directive = UnicodeString(ExtensionMark) + Key;        UnicodeString Value = ExtensionLine.SubString(P + 1, ExtensionLine.Length() - P).Trim();        bool KnownKey = true;        if (Key == ExtensionNameDirective)        {          Name = WinConfiguration->ExtensionStringTranslation(Id, Value);        }        else if (Key == ExtensionCommandDirective)        {          Command = Value;        }        else if (Key == L"require")        {          UnicodeString DependencyVersion = Value;          UnicodeString Dependency = CutToChar(Value, L' ', true).LowerCase();          Value = Value.Trim();          bool Failed;          if (Dependency == L"winscp")          {            int Version = StrToCompoundVersion(Value);            Failed = (Version > WinConfiguration->CompoundVersion);          }          else if (Dependency == L".net")          {            Failed = (CompareVersion(Value, GetNetVersionStr()) > 0);          }          else if (Dependency == L"powershell")          {            Failed = (CompareVersion(Value, GetPowerShellVersionStr()) > 0);          }          else if (Dependency == L"windows")          {            Failed = (CompareVersion(Value, WindowsVersion()) > 0);          }          else          {            Failed = true;          }          if (Failed)          {            throw Exception(MainInstructions(FMTLOAD(EXTENSION_DEPENDENCY_ERROR, (DependencyVersion))));          }        }        else if (Key == L"side")        {          if (SameText(Value, L"Local"))          {            Params |= ccLocal;          }          else if (SameText(Value, L"Remote"))          {            Params &= ~ccLocal;          }          else          {            throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive))));          }        }        else if (Key == L"flag")        {          if (SameText(Value, L"ApplyToDirectories"))          {            Params |= ccApplyToDirectories;          }          else if (SameText(Value, L"Recursive"))          {            Params |= ccRecursive;          }          else if (SameText(Value, L"ShowResults"))          {            Params |= ccShowResults;          }          else if (SameText(Value, L"CopyResults"))          {            Params |= ccCopyResults;          }          else if (SameText(Value, L"RemoteFiles"))          {            Params |= ccRemoteFiles;          }          else if (SameText(Value, L"ShowResultsInMsgBox"))          {            Params |= ccShowResultsInMsgBox;          }          else          {            throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive))));          }        }        else if (Key == L"shortcut")        {          TShortCut AShortCut = TextToShortCut(Value);          if (IsCustomShortCut(AShortCut))          {            ShortCut = AShortCut;            FShortCutOriginal = AShortCut;          }          else          {            throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive))));          }        }        else if (Key == L"option")        {          TOption Option;          if (!ParseOption(Value, Option, ExtensionBaseName) ||              (Option.IsControl && (OptionIds.find(Option.Id.LowerCase()) != OptionIds.end())))          {            throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_ERROR, (Value, Directive))));          }          else          {            FOptions.push_back(Option);            if (!Option.IsControl)            {              OptionIds.insert(Option.Id.LowerCase());            }          }        }        else if (Key == L"description")        {          Description = WinConfiguration->ExtensionStringTranslation(Id, Value);        }        else if (Key == L"author")        {          // noop        }        else if (Key == L"version")        {          // noop        }        else if (Key == L"homepage")        {          HomePage = Value;        }        else if (Key == L"optionspage")        {          OptionsPage = Value;        }        else if (Key == L"source")        {          // noop        }        else        {          KnownKey = false;        }        if (KnownKey)        {          AnythingFound = true;        }      }      ExtensionLine = L"";    }  }  if (!AnythingFound)  {    throw Exception(MainInstructions(LoadStr(EXTENSION_NOT_FOUND)));  }  if (Name.IsEmpty())  {    throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_MISSING, (UnicodeString(ExtensionMark) + ExtensionNameDirective))));  }  if (Command.IsEmpty())  {    throw Exception(MainInstructions(FMTLOAD(EXTENSION_DIRECTIVE_MISSING, (UnicodeString(ExtensionMark) + ExtensionCommandDirective))));  }}//---------------------------------------------------------------------------bool __fastcall TCustomCommandType::ParseOption(const UnicodeString & Value, TOption & Option, const UnicodeString & ExtensionBaseName){  UnicodeString Buf = Value;  UnicodeString KindName;  bool Result = CutTokenEx(Buf, Option.Id);  if (Result)  {    Option.Flags = 0;    UnicodeString FlagName;    while (CutTokenEx(Buf, FlagName) && (FlagName.SubString(1, 1) == L"-"))    {      FlagName = FlagName.LowerCase();      if (FlagName == L"-run")      {        Option.Flags |= ofRun;      }      else if (FlagName == L"-config")      {        Option.Flags |= ofConfig;      }      else if (FlagName == L"-site")      {        Option.Flags |= ofSite;      }      else      {        Result = false;      }    }    if ((Option.Flags & (ofRun | ofConfig)) == 0)    {      Option.Flags |= ofConfig;    }    KindName = FlagName;    if (Result)    {      UnicodeString DefaultCaption;      UnicodeString DefaultDefault;      TOption::TParams DefaultParams;      KindName = KindName.LowerCase();      if (KindName == L"label")      {        Option.Kind = okLabel;        Result = !Option.IsControl;      }      else if (KindName == L"link")      {        Option.Kind = okLink;        Result = !Option.IsControl;      }      else if (KindName == L"separator")      {        Option.Kind = okSeparator;        Result = !Option.IsControl;      }      else if (KindName == L"group")      {        Option.Kind = okGroup;        Result = !Option.IsControl;      }      else if (KindName == L"textbox")      {        Option.Kind = okTextBox;        Result = Option.IsControl;      }      else if (KindName == L"file")      {        Option.Kind = okFile;        Result = Option.IsControl;      }      else if (KindName == L"sessionlogfile")      {        Option.Kind = okFile;        Result = Option.IsControl;        DefaultCaption = LoadStr(EXTENSION_SESSIONLOG_FILE);        Option.FileCaption = LoadStr(EXTENSION_SESSIONLOG_CAPTION);        Option.FileFilter = LoadStr(EXTENSION_SESSIONLOG_FILTER);        // Similar to TConfiguration::GetDefaultLogFileName        Option.FileInitial = L"%TEMP%\\" + ExtensionBaseName + L".log";        Option.FileExt = L"log";      }      else if (KindName == L"dropdownlist")      {        Option.Kind = okDropDownList;        Result = Option.IsControl;      }      else if (KindName == L"combobox")      {        Option.Kind = okComboBox;        Result = Option.IsControl;      }      else if (KindName == L"checkbox")      {        Option.Kind = okCheckBox;        Result = Option.IsControl;      }      else if (KindName == L"pausecheckbox")      {        Option.Kind = okCheckBox;        Result = Option.IsControl;        DefaultCaption = LoadStr(EXTENSION_PAUSE_CHECKBOX);        DefaultDefault = L"-pause";        DefaultParams.push_back(L"-pause");      }      else      {        Option.Kind = okUnknown;      }      if ((Option.Kind != okUnknown) &&          (Option.Kind != okSeparator))      {        Result =          CutTokenEx(Buf, Option.Caption);        if (!Result && !DefaultCaption.IsEmpty())        {          Option.Caption = DefaultCaption;          Result = true;        }        if (Result)        {          Option.Caption = WinConfiguration->ExtensionStringTranslation(Id, Option.Caption);          if (CutTokenEx(Buf, Option.Default))          {            UnicodeString Param;            while (CutTokenEx(Buf, Param))            {              Option.Params.push_back(Param);            }          }          else          {            Option.Default = DefaultDefault;          }          if (Option.Params.size() == 0)          {            Option.Params = DefaultParams;          }        }      }    }  }  return Result;}//---------------------------------------------------------------------------int __fastcall TCustomCommandType::GetOptionsCount() const{  return FOptions.size();}//---------------------------------------------------------------------------const TCustomCommandType::TOption & __fastcall TCustomCommandType::GetOption(int Index) const{  return FOptions[Index];}//---------------------------------------------------------------------------UnicodeString __fastcall TCustomCommandType::GetOptionKey(  const TCustomCommandType::TOption & Option, const UnicodeString & Site) const{  UnicodeString Result = Id + L"\\" + Option.Id;  if (FLAGSET(Option.Flags, ofSite))  {    Result += L"\\" + Site;  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TCustomCommandType::AnyOptionWithFlag(unsigned int Flag) const{  bool Result = false;  for (int Index = 0; !Result && (Index < OptionsCount); Index++)  {    const TCustomCommandType::TOption & Option = GetOption(Index);    Result = FLAGSET(Option.Flags, Flag);  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TCustomCommandType::GetCommandWithExpandedOptions(  TStrings * CustomCommandOptions, const UnicodeString & Site) const{  UnicodeString Result = Command;  for (int Index = 0; Index < OptionsCount; Index++)  {    const TCustomCommandType::TOption & Option = GetOption(Index);    if (Option.IsControl)    {      UnicodeString OptionKey = GetOptionKey(Option, Site);      UnicodeString OptionValue;      if (CustomCommandOptions->IndexOfName(OptionKey) >= 0)      {        OptionValue = CustomCommandOptions->Values[OptionKey];      }      else      {        OptionValue = Option.Default;      }      UnicodeString OptionCommand = GetOptionCommand(Option, OptionValue);      OptionCommand = TCustomCommand::Escape(OptionCommand);      Result = ReplaceText(Result, FORMAT(L"%%%s%%", (Option.Id)), OptionCommand);    }  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TCustomCommandType::GetOptionCommand(const TOption & Option, const UnicodeString & Value) const{  UnicodeString Result = Value;  switch (Option.Kind)  {    case okUnknown:    case okTextBox:    case okDropDownList:    case okComboBox:    case okCheckBox:      // noop      break;    case okFile:      Result = ExpandEnvironmentVariables(Result);      break;    case okLabel:    case okLink:    case okSeparator:    case okGroup:    default:      DebugFail();  }  return Result;}//---------------------------------------------------------------------------bool __fastcall TCustomCommandType::HasCustomShortCut() const{  return (ShortCut != FShortCutOriginal);}//---------------------------------------------------------------------------//---------------------------------------------------------------------------bool __fastcall TCustomCommandType::TOption::GetIsControl() const{  return (Id != L"-");}//---------------------------------------------------------------------------bool TCustomCommandType::TOption::HasPatterns(TCustomCommand * CustomCommandForOptions) const{  bool CanHavePatterns;  switch (Kind)  {    case okTextBox:    case okFile:      CanHavePatterns = true;      break;    default:      CanHavePatterns = false;      break;  }  bool Result =    CanHavePatterns &&    FLAGSET(Flags, TCustomCommandType::ofRun) &&    FLAGCLEAR(Flags, TCustomCommandType::ofConfig) &&    CustomCommandForOptions->HasAnyPatterns(Default);  return Result;}//---------------------------------------------------------------------------bool TCustomCommandType::TOption::operator==(const TCustomCommandType::TOption & Other) const{  // needed by vector<> but probably never used  return    (Id == Other.Id) &&    (Flags == Other.Flags) &&    (Kind == Other.Kind) &&    (Caption == Other.Caption) &&    (Default == Other.Default) &&    (Params == Other.Params) &&    (FileCaption == Other.FileCaption) &&    (FileFilter == Other.FileFilter) &&    (FileInitial == Other.FileInitial) &&    (FileExt == Other.FileExt);}//---------------------------------------------------------------------------//---------------------------------------------------------------------------__fastcall TCustomCommandList::TCustomCommandList(){  FCommands = new TList();  FModified = false;}//---------------------------------------------------------------------------__fastcall TCustomCommandList::~TCustomCommandList(){  Clear();  delete FCommands;}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Reset(){  FModified = false;}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Modify(){  FModified = true;}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Load(THierarchicalStorage * Storage){  Clear();  if (Storage->OpenSubKey(L"CustomCommands", false))  {    TStrings * Names = new TStringList();    try    {      Storage->ReadValues(Names, true);      for (int Index = 0; Index < Names->Count; Index++)      {        TCustomCommandType * Command = new TCustomCommandType();        Command->Name = Names->Names[Index];        Command->Command = Names->Values[Names->Names[Index]];        FCommands->Add(Command);      }      Storage->CloseSubKey();    }    __finally    {      delete Names;    }  }  if (Storage->OpenSubKey(L"CustomCommandsParams", false))  {    for (int Index = 0; Index < FCommands->Count; Index++)    {      TCustomCommandType * Command = GetCommand(Index);      Command->Params = Storage->ReadInteger(Command->Name, Command->Params);    }    Storage->CloseSubKey();  }  if (Storage->OpenSubKey(L"CustomCommandsShortCuts", false))  {    for (int Index = 0; Index < FCommands->Count; Index++)    {      TCustomCommandType * Command = GetCommand(Index);      Command->ShortCut = (Word)Storage->ReadInteger(Command->Name, Command->ShortCut);    }    Storage->CloseSubKey();  }  Reset();}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Save(THierarchicalStorage * Storage){  if (Storage->OpenSubKey(L"CustomCommands", true))  {    Storage->ClearValues();    for (int Index = 0; Index < FCommands->Count; Index++)    {      const TCustomCommandType * Command = Commands[Index];      Storage->WriteString(Command->Name, Command->Command);    }    Storage->CloseSubKey();  }  if (Storage->OpenSubKey(L"CustomCommandsParams", true))  {    Storage->ClearValues();    for (int Index = 0; Index < FCommands->Count; Index++)    {      const TCustomCommandType * Command = Commands[Index];      Storage->WriteInteger(Command->Name, Command->Params);    }    Storage->CloseSubKey();  }  if (Storage->OpenSubKey(L"CustomCommandsShortCuts", true))  {    Storage->ClearValues();    for (int Index = 0; Index < FCommands->Count; Index++)    {      const TCustomCommandType * Command = Commands[Index];      if (Command->ShortCut != 0)      {        Storage->WriteInteger(Command->Name, Command->ShortCut);      }    }    Storage->CloseSubKey();  }}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Clear(){  for (int Index = 0; Index < FCommands->Count; Index++)  {    delete Commands[Index];  }  FCommands->Clear();}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Add(const UnicodeString Name,  const UnicodeString ACommand, int Params){  TCustomCommandType * Command = new TCustomCommandType();  Command->Name = Name;  Command->Command = ACommand;  Command->Params = Params;  Add(Command);}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Add(TCustomCommandType * Command){  Insert(Count, Command);}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Insert(int Index, TCustomCommandType * Command){  FCommands->Insert(Index, Command);  Modify();}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Change(int Index, TCustomCommandType * ACommand){  TCustomCommandType * Command = GetCommand(Index);  if (!Command->Equals(ACommand))  {    delete Command;    FCommands->Items[Index] = ACommand;    Modify();  }  else  {    delete ACommand;  }}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Move(int CurIndex, int NewIndex){  if (CurIndex != NewIndex)  {    FCommands->Move(CurIndex, NewIndex);    Modify();  }}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Delete(int Index){  DebugAssert((Index >= 0) && (Index < Count));  delete GetCommand(Index);  FCommands->Delete(Index);  Modify();}//---------------------------------------------------------------------------class TCustomCommandCompareFunc : public TCppInterfacedObject<TListSortCompareFunc>{public:  TCustomCommandCompareFunc(TStrings * Ids)  {    FIds = Ids;  }  virtual int __fastcall Invoke(void * Item1, void * Item2)  {    TCustomCommandType * CustomCommand1 = static_cast<TCustomCommandType *>(Item1);    TCustomCommandType * CustomCommand2 = static_cast<TCustomCommandType *>(Item2);    int Index1 = FIds->IndexOf(CustomCommand1->Id);    int Index2 = FIds->IndexOf(CustomCommand2->Id);    int Result;    // new items to the end    if ((Index1 < 0) && (Index2 >= 0))    {      Result = 1;    }    else if ((Index2 < 0) && (Index1 >= 0))    {      Result = -1;    }    // fallback to comparing by name    else if ((Index1 < 0) && (Index2 < 0))    {      Result = TComparer__1<UnicodeString>::Default()->Compare(CustomCommand1->Name, CustomCommand2->Name);    }    else    {      Result = TComparer__1<int>::Default()->Compare(Index1, Index2);    }    return Result;  }private:  TStrings * FIds;};//---------------------------------------------------------------------------void __fastcall TCustomCommandList::SortBy(TStrings * Ids){  TCustomCommandCompareFunc * Func = new TCustomCommandCompareFunc(Ids);  FCommands->SortList(Func);}//---------------------------------------------------------------------------int __fastcall TCustomCommandList::GetCount() const{  return FCommands->Count;}//---------------------------------------------------------------------------const TCustomCommandType * __fastcall TCustomCommandList::GetConstCommand(int Index) const{  return static_cast<TCustomCommandType *>(FCommands->Items[Index]);}//---------------------------------------------------------------------------TCustomCommandType * __fastcall TCustomCommandList::GetCommand(int Index){  return static_cast<TCustomCommandType *>(FCommands->Items[Index]);}//---------------------------------------------------------------------------bool __fastcall TCustomCommandList::Equals(const TCustomCommandList * Other) const{  bool Result = (Count == Other->Count);  for (int Index = 0; Result && (Index < Count); Index++)  {    Result = Commands[Index]->Equals(Other->Commands[Index]);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::Assign(const TCustomCommandList * Other){  Clear();  for (int Index = 0; Index < Other->Count; Index++)  {    Add(new TCustomCommandType(*Other->Commands[Index]));  }  // there should be comparison of with the assigned list, be we rely on caller  // to do it instead (TGUIConfiguration::SetCopyParamList)  Modify();}//---------------------------------------------------------------------------const TCustomCommandType * TCustomCommandList::Find(const UnicodeString Name) const{  for (int Index = 0; Index < FCommands->Count; Index++)  {    if (Commands[Index]->Name == Name)    {      return Commands[Index];    }  }  return NULL;}//---------------------------------------------------------------------------const TCustomCommandType * TCustomCommandList::Find(TShortCut ShortCut) const{  for (int Index = 0; Index < FCommands->Count; Index++)  {    if (Commands[Index]->ShortCut == ShortCut)    {      return Commands[Index];    }  }  return NULL;}//---------------------------------------------------------------------------int TCustomCommandList::FindIndexByFileName(const UnicodeString & FileName) const{  for (int Index = 0; Index < FCommands->Count; Index++)  {    if (IsPathToSameFile(Commands[Index]->FileName, FileName))    {      return Index;    }  }  return -1;}//---------------------------------------------------------------------------void __fastcall TCustomCommandList::ShortCuts(TShortCuts & ShortCuts) const{  for (int Index = 0; Index < FCommands->Count; Index++)  {    const TCustomCommandType * Command = Commands[Index];    if (Command->ShortCut != 0)    {      ShortCuts.Add(Command->ShortCut);    }  }}
 |