| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957 | //---------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include <StrUtils.hpp>#include <Common.h>#include <math.h>#include "Preferences.h"#include "Custom.h"#include <CoreMain.h>#include <Terminal.h>#include <Bookmarks.h>#include "VCLCommon.h"#include "GUITools.h"#include "Tools.h"#include "TextsCore.h"#include "TextsWin.h"#include "HelpWin.h"#include "WinInterface.h"#include "WinConfiguration.h"#include "Setup.h"#include "ProgParams.h"#include "Http.h"#include "TerminalManager.h"//---------------------------------------------------------------------#pragma link "CopyParams"#pragma link "UpDownEdit"#pragma link "ComboEdit"#pragma link "HistoryComboBox"#pragma link "PathLabel"#ifndef NO_RESOURCES#pragma resource "*.dfm"#endif//---------------------------------------------------------------------const int BelowNormalLogLevels = 1;//---------------------------------------------------------------------bool __fastcall DoPreferencesDialog(TPreferencesMode APreferencesMode,  TPreferencesDialogData * DialogData){  bool Result;  TPreferencesDialog * PreferencesDialog =    new TPreferencesDialog(GetFormOwner(), APreferencesMode);  try  {    Result = PreferencesDialog->Execute(DialogData);    if (Result)    {      CheckConfigurationForceSave();      Configuration->SaveExplicit();    }  }  __finally  {    delete PreferencesDialog;  }  return Result;}//---------------------------------------------------------------------__fastcall TPreferencesDialog::TPreferencesDialog(  TComponent* AOwner, TPreferencesMode PreferencesMode)  : TForm(AOwner){  SetCorrectFormParent(this);  FNoUpdate = 0;  FPreferencesMode = PreferencesMode;  FEditorFont.reset(new TFont());  FEditorFont->Color = clWindowText;  FPanelFont.reset(new TFont());  FPanelFont->Color = clWindowText;  // currently useless  FAfterFilenameEditDialog = false;  FCustomCommandList = new TCustomCommandList();  FCustomCommandChanging = false;  FExtensionList = new TCustomCommandList();  FListViewDragDest = -1;  FCopyParamList = new TCopyParamList();  FEditorList = new TEditorList();  FAutomaticUpdatesPossible = IsInstalled();  FCustomCommandsHintItem = NULL;  FAddedExtensions.reset(CreateSortedStringList());  FCustomCommandOptions.reset(new TStringList());  UseSystemSettings(this);  FCustomCommandsScrollOnDragOver = new TListViewScrollOnDragOver(CustomCommandsView, true);  FixListColumnWidth(CustomCommandsView, -1);  FCopyParamScrollOnDragOver = new TListViewScrollOnDragOver(CopyParamListView, true);  FixListColumnWidth(CopyParamListView, -1);  FEditorScrollOnDragOver = new TListViewScrollOnDragOver(EditorListView3, true);  FixListColumnWidth(EditorListView3, -1);  FOrigCustomCommandsViewWindowProc = CustomCommandsView->WindowProc;  CustomCommandsView->WindowProc = CustomCommandsViewWindowProc;  ComboAutoSwitchInitialize(UpdatesBetaVersionsCombo);  EnableControl(UpdatesBetaVersionsCombo, !WinConfiguration->IsBeta && !IsUWP());  EnableControl(UpdatesBetaVersionsLabel, UpdatesBetaVersionsCombo->Enabled);  HintLabel(LogFileNameHintText, LoadStr(LOG_FILE_HINT3));  HintLabel(ActionsLogFileNameHintText, LoadStr(LOG_FILE_HINT3));  HintLabel(ShellIconsText2);  HotTrackLabel(CopyParamLabel);  HintLabel(PuttyPathHintText, LoadStr(PUTTY_PATTERNS_HINT2));  EditorEncodingCombo->Items->Add(DefaultEncodingName());  EditorEncodingCombo->Items->Add(LoadStr(UTF8_NAME));  std::unique_ptr<TStrings> Workspaces(StoredSessions->GetWorkspaces());  AutoWorkspaceCombo->Items->AddStrings(Workspaces.get());  AutoSaveWorkspacePasswordsCheck->Caption = LoadStr(SAVE_WORKSPACE_PASSWORDS);  PuttyRegistryStorageKeyEdit->Items->Add(OriginalPuttyRegistryStorageKey);  PuttyRegistryStorageKeyEdit->Items->Add(KittyRegistryStorageKey);  MenuButton(RegisterAsUrlHandlersButton);  MenuButton(EditorFontColorButton);  MenuButton(EditorBackgroundColorButton);  LoadDialogImage(CommanderInterfacePicture, L"Commander");  LoadDialogImage(ExplorerInterfacePicture, L"Explorer");  LinkLabel(UpdatesLink);  LinkAppLabel(BackgroundConfirmationsLink);  AutomaticIniFileStorageLabel->Caption = ExpandEnvironmentVariables(Configuration->GetAutomaticIniFileStorageName(false));  HideComponentsPanel(this);}//---------------------------------------------------------------------------__fastcall TPreferencesDialog::~TPreferencesDialog(){  SAFE_DESTROY(FEditorScrollOnDragOver);  SAFE_DESTROY(FCopyParamScrollOnDragOver);  SAFE_DESTROY(FCustomCommandsScrollOnDragOver);  delete FCustomCommandList;  FCustomCommandList = NULL;  delete FExtensionList;  FExtensionList = NULL;  delete FCopyParamList;  FCopyParamList = NULL;  delete FEditorList;  FEditorList = NULL;}//---------------------------------------------------------------------bool __fastcall TPreferencesDialog::Execute(TPreferencesDialogData * DialogData){  PuttyPathEdit->Items = CustomWinConfiguration->History[L"PuttyPath"];  FDialogData = DialogData;  bool Result = (ShowModal() == DefaultResult(this));  if (Result)  {    SaveConfiguration();    CustomWinConfiguration->History[L"PuttyPath"] = PuttyPathEdit->Items;  }  else  {    for (int Index = 0; Index < FAddedExtensions->Count; Index++)    {      DeleteFile(ApiPath(FAddedExtensions->Strings[Index]));    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::LoadLanguages(){  if (!FLanguagesLoaded)  {    LanguagesView->Items->Clear();    LCID Locale = GUIConfiguration->Locale;    if (Locale == NULL)    {      DebugAssert(GUIConfiguration->AppliedLocale == WinConfiguration->DefaultLocale);      Locale = WinConfiguration->DefaultLocale;    }    TObjectList * Locales = GUIConfiguration->Locales;    for (int Index = 0; Index < Locales->Count; Index++)    {      TListItem * Item = LanguagesView->Items->Add();      TLocaleInfo * LocaleInfo = DebugNotNull(dynamic_cast<TLocaleInfo *>(Locales->Items[Index]));      Item->Caption = LocaleInfo->Name;      Item->Data = LocaleInfo;      Item->Focused = (LocaleInfo->Locale == Locale);      Item->Selected = Item->Focused;    }    AutoSizeListColumnsWidth(LanguagesView);    FLanguagesLoaded = true;  }}//---------------------------------------------------------------------------TTabSheet * __fastcall TPreferencesDialog::FindPageForTreeNode(TTreeNode * Node){  for (int pi = 0; pi < PageControl->PageCount; pi++)  {    TTabSheet * Sheet = PageControl->Pages[pi];    if (Sheet->Tag == Node->SelectedIndex)    {      return Sheet;    }  }  DebugFail();  return NULL;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PrepareNavigationTree(TTreeView * Tree){  Tree->FullExpand();  int i = 0;  while (i < Tree->Items->Count)  {    TTreeNode * Node = Tree->Items->Item[i];    TTabSheet * Sheet = FindPageForTreeNode(Node);    if (DebugNotNull(Sheet))    {      if (Sheet->Enabled)      {        // gets called multiple times occasionally        // (e.g. when called from upload dialog invoked by /upload)        if (!Sheet->Caption.IsEmpty())        {          Node->Text = Sheet->Caption;          Sheet->Caption = L"";        }      }      else      {        Tree->Items->Delete(Node);        i--;      }    }    i++;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::LoadConfiguration(){  FNoUpdate++;  try  {    #define BOOLPROP(PROP) PROP ## Check->Checked = WinConfiguration->PROP;    BOOLPROP(DefaultDirIsHome);    BOOLPROP(PreservePanelState);    BOOLPROP(DeleteToRecycleBin);    BOOLPROP(DDWarnLackOfTempSpace);    BOOLPROP(ShowHiddenFiles);    BOOLPROP(RenameWholeName);    BOOLPROP(ShowInaccesibleDirectories);    BOOLPROP(CopyOnDoubleClickConfirmation);    BOOLPROP(ConfirmTransferring);    BOOLPROP(ConfirmOverwriting);    BOOLPROP(ConfirmResume);    BOOLPROP(ConfirmDeleting);    BOOLPROP(ConfirmRecycling);    BOOLPROP(ConfirmExitOnCompletion);    BOOLPROP(ConfirmCommandSession);    BOOLPROP(ContinueOnError);    BOOLPROP(SynchronizeSummary);    BOOLPROP(BeepOnFinish);    BOOLPROP(TemporaryDirectoryAppendSession);    BOOLPROP(TemporaryDirectoryAppendPath);    BOOLPROP(TemporaryDirectoryDeterministic);    BOOLPROP(TemporaryDirectoryCleanup);    BOOLPROP(ConfirmTemporaryDirectoryCleanup);    BOOLPROP(FullRowSelect);    BOOLPROP(NaturalOrderNumericalSorting);    ConfirmClosingSessionCheck2->Checked = WinConfiguration->ConfirmClosingSession;    if (WinConfiguration->DDTransferConfirmation == asAuto)    {      // allow grayed state only initially,      // once the off state is confirmed, never allow returning      // to the undefined state      DDTransferConfirmationCheck2->AllowGrayed = true;    }    CheckBoxAutoSwitchLoad(DDTransferConfirmationCheck2, WinConfiguration->DDTransferConfirmation);    BeepOnFinishAfterEdit->AsInteger =      int(static_cast<double>(GUIConfiguration->BeepOnFinishAfter) * SecsPerDay);    BOOLPROP(BalloonNotifications);    DDFakeFileEnabledButton->Checked = WinConfiguration->DDFakeFile;    DDFakeFileDisabledButton->Checked = !DDFakeFileEnabledButton->Checked;    if (WinConfiguration->DDTemporaryDirectory.IsEmpty())    {      DDSystemTemporaryDirectoryButton->Checked = true;      DDTemporaryDirectoryEdit->Text = SystemTemporaryDirectory();    }    else    {      DDCustomTemporaryDirectoryButton->Checked = true;      DDTemporaryDirectoryEdit->Text = WinConfiguration->DDTemporaryDirectory;    }    // Commander    if (WinConfiguration->ScpCommander.NortonLikeMode == nlOff)    {      NortonLikeModeCombo->ItemIndex = 2;    }    else if (WinConfiguration->ScpCommander.NortonLikeMode == nlKeyboard)    {      NortonLikeModeCombo->ItemIndex = 1;    }    else    {      NortonLikeModeCombo->ItemIndex = 0;    }    SwappedPanelsCheck->Checked =      WinConfiguration->ScpCommander.SwappedPanels;    TreeOnLeftCheck->Checked = WinConfiguration->ScpCommander.TreeOnLeft;    ExplorerKeyboardShortcutsCombo->ItemIndex =      WinConfiguration->ScpCommander.ExplorerKeyboardShortcuts ? 1 : 0;    BOOLPROP(UseLocationProfiles);    CompareByTimeCheck->Checked = WinConfiguration->ScpCommander.CompareByTime;    CompareBySizeCheck->Checked = WinConfiguration->ScpCommander.CompareBySize;    // Local panel    PreserveLocalDirectoryCheck->Checked =      WinConfiguration->ScpCommander.PreserveLocalDirectory;    SystemContextMenuCheck->Checked =      WinConfiguration->ScpCommander.SystemContextMenu;    // Explorer    ShowFullAddressCheck->Checked =      WinConfiguration->ScpExplorer.ShowFullAddress;    // select none when stNul    RegistryStorageButton->Checked = (Configuration->Storage == stRegistry);    AutomaticIniFileStorageButton->Checked = (Configuration->Storage == stIniFile) && Configuration->CustomIniFileStorageName.IsEmpty();    CustomIniFileStorageButton->Checked = (Configuration->Storage == stIniFile) && !Configuration->CustomIniFileStorageName.IsEmpty();    CustomIniFileStorageEdit->Text = Configuration->CustomIniFileStorageName;    if (Configuration->CustomIniFileStorageName.IsEmpty())    {      CustomIniFileStorageEdit->Text = Configuration->GetDefaultIniFileExportPath();    }    else    {      CustomIniFileStorageEdit->Text = Configuration->CustomIniFileStorageName;    }    FCustomIniFileStorageName = GetCustomIniFileStorageName();    RandomSeedFileEdit->Text = Configuration->RandomSeedFile;    // editor    EditorWordWrapCheck->Checked = WinConfiguration->Editor.WordWrap;    EditorTabSizeEdit->AsInteger = WinConfiguration->Editor.TabSize;    if (WinConfiguration->Editor.Encoding == CP_UTF8)    {      EditorEncodingCombo->ItemIndex = 1;    }    else    {      EditorEncodingCombo->ItemIndex = 0;    }    TWinConfiguration::RestoreFont(WinConfiguration->Editor.Font, FEditorFont.get());    FEditorFont->Color = WinConfiguration->Editor.FontColor;    FEditorBackgroundColor = WinConfiguration->Editor.BackgroundColor;    (*FEditorList) = *WinConfiguration->EditorList;    UpdateEditorListView();    FCopyParams = GUIConfiguration->DefaultCopyParam;    ResumeOnButton->Checked = GUIConfiguration->DefaultCopyParam.ResumeSupport == rsOn;    ResumeSmartButton->Checked = GUIConfiguration->DefaultCopyParam.ResumeSupport == rsSmart;    ResumeOffButton->Checked = GUIConfiguration->DefaultCopyParam.ResumeSupport == rsOff;    ResumeThresholdEdit->Value = GUIConfiguration->DefaultCopyParam.ResumeThreshold / 1024;    SessionReopenAutoCheck->Checked = (Configuration->SessionReopenAuto > 0);    SessionReopenAutoEdit->Value = (Configuration->SessionReopenAuto > 0 ?      (Configuration->SessionReopenAuto / MSecsPerSec) : 5);    SessionReopenAutoIdleCheck->Checked = (GUIConfiguration->SessionReopenAutoIdle > 0);    SessionReopenAutoIdleEdit->Value = (GUIConfiguration->SessionReopenAutoIdle > 0 ?      (GUIConfiguration->SessionReopenAutoIdle / MSecsPerSec) : 5);    SessionReopenAutoStallCheck->Checked = (Configuration->SessionReopenAutoStall > 0);    SessionReopenAutoStallEdit->Value = (Configuration->SessionReopenAutoStall > 0 ?      (Configuration->SessionReopenAutoStall / MSecsPerSec) : SecsPerMin);    SessionReopenTimeoutEdit->Value = (Configuration->SessionReopenTimeout / MSecsPerSec);    GeneralSheet->Enabled = WinConfiguration->ExpertMode;    ExplorerSheet->Enabled = WinConfiguration->ExpertMode;    CommanderSheet->Enabled = WinConfiguration->ExpertMode;    EditorSheet->Enabled = WinConfiguration->ExpertMode && !WinConfiguration->DisableOpenEdit;    StorageGroup->Visible = WinConfiguration->ExpertMode;    RandomSeedFileLabel->Visible = WinConfiguration->ExpertMode;    RandomSeedFileEdit->Visible = WinConfiguration->ExpertMode;    FCustomCommandList->Assign(WinConfiguration->CustomCommandList);    FExtensionList->Assign(WinConfiguration->ExtensionList);    UpdateCustomCommandsView();    FCustomCommandOptions->Assign(WinConfiguration->CustomCommandOptions);    PuttyPathEdit->Text = GUIConfiguration->PuttyPath;    PuttyPasswordCheck2->Checked = GUIConfiguration->PuttyPassword;    AutoOpenInPuttyCheck->Checked = WinConfiguration->AutoOpenInPutty;    TelnetForFtpInPuttyCheck->Checked = WinConfiguration->TelnetForFtpInPutty;    SelectPuttyRegistryStorageKey(GUIConfiguration->PuttyRegistryStorageKey);    // Queue    QueueTransferLimitEdit->AsInteger = GUIConfiguration->QueueTransfersLimit;    EnableQueueByDefaultCheck->Checked = WinConfiguration->EnableQueueByDefault;    QueueAutoPopupCheck->Checked = GUIConfiguration->QueueAutoPopup;    QueueCheck->Checked = GUIConfiguration->DefaultCopyParam.Queue;    QueueParallelCheck->Checked = GUIConfiguration->DefaultCopyParam.QueueParallel;    QueueNoConfirmationCheck->Checked = GUIConfiguration->DefaultCopyParam.QueueNoConfirmation;    if (!GUIConfiguration->QueueKeepDoneItems)    {      QueueKeepDoneItemsForCombo->ItemIndex = 0;    }    else if (GUIConfiguration->QueueKeepDoneItemsFor < 0)    {      QueueKeepDoneItemsForCombo->ItemIndex = 5;    }    else if (GUIConfiguration->QueueKeepDoneItemsFor <= 15)    {      QueueKeepDoneItemsForCombo->ItemIndex = 1;    }    else if (GUIConfiguration->QueueKeepDoneItemsFor <= 60)    {      QueueKeepDoneItemsForCombo->ItemIndex = 2;    }    else if (GUIConfiguration->QueueKeepDoneItemsFor <= 15 * 60)    {      QueueKeepDoneItemsForCombo->ItemIndex = 3;    }    else if (GUIConfiguration->QueueKeepDoneItemsFor <= 60 * 60)    {      QueueKeepDoneItemsForCombo->ItemIndex = 4;    }    if (WinConfiguration->QueueView.Show == qvShow)    {      QueueViewShowButton->Checked = true;    }    else if (WinConfiguration->QueueView.Show == qvHideWhenEmpty)    {      QueueViewHideWhenEmptyButton->Checked = true;    }    else    {      QueueViewHideButton->Checked = true;    }    // window    AutoSaveWorkspaceCheck->Checked = WinConfiguration->AutoSaveWorkspace;    AutoWorkspaceCombo->Text =      DefaultStr(WinConfiguration->AutoWorkspace,        // It will rarely happen that LastWorkspace is set, while AutoWorkspace not.        // It can happen only when user saved workspace before opening the Preferences        // dialog for the first time        DefaultStr(WinConfiguration->LastWorkspace, LoadStr(NEW_WORKSPACE)));    AutoSaveWorkspacePasswordsCheck->Checked = WinConfiguration->AutoSaveWorkspacePasswords;    if (WinConfiguration->PathInCaption == picFull)    {      PathInCaptionFullButton->Checked = true;    }    else if (WinConfiguration->PathInCaption == picShort)    {      PathInCaptionShortButton->Checked = true;    }    else    {      PathInCaptionNoneButton->Checked = true;    }    BOOLPROP(MinimizeToTray);    BOOLPROP(ExternalSessionInExistingInstance);    BOOLPROP(KeepOpenWhenNoSession);    BOOLPROP(ShowTips);    // panels    DoubleClickActionCombo->ItemIndex = WinConfiguration->DoubleClickAction;    BOOLPROP(AutoReadDirectoryAfterOp);    BOOLPROP(RefreshRemotePanel);    RefreshRemotePanelIntervalEdit->Value =      int(static_cast<double>(WinConfiguration->RefreshRemotePanelInterval) * SecsPerDay);    switch (WinConfiguration->FormatSizeBytes)    {      case fbNone:        FormatSizeBytesCombo->ItemIndex = 0;        break;      case fbKilobytes:        FormatSizeBytesCombo->ItemIndex = 1;        break;      case fbShort:        FormatSizeBytesCombo->ItemIndex = 2;        break;      default:        DebugFail();    }    bool CustomPanelFont = !WinConfiguration->PanelFont.FontName.IsEmpty();    PanelFontCheck->Checked = CustomPanelFont;    if (CustomPanelFont)    {      TWinConfiguration::RestoreFont(WinConfiguration->PanelFont, FPanelFont.get());    }    else    {      // Default to system icon font, when starting customization      FPanelFont->Assign(Screen->IconFont);    }    // updates    TUpdatesConfiguration Updates = WinConfiguration->Updates;    if (int(Updates.Period) <= 0)    {      UpdatesPeriodCombo->ItemIndex = 0;    }    else if (int(Updates.Period) <= 1)    {      UpdatesPeriodCombo->ItemIndex = 1;    }    else if (int(Updates.Period) <= 7)    {      UpdatesPeriodCombo->ItemIndex = 2;    }    else    {      UpdatesPeriodCombo->ItemIndex = 3;    }    UpdatesShowOnStartup->Checked = Updates.ShowOnStartup;    UpdatesAuthenticationEmailEdit->Text = Updates.AuthenticationEmail;    FVerifiedUpdatesAuthenticationEmail = UpdatesAuthenticationEmailEdit->Text;    CollectUsageCheck->Checked = Configuration->CollectUsage;    ComboAutoSwitchLoad(UpdatesBetaVersionsCombo, Updates.BetaVersions);    switch (Updates.ConnectionType)    {      case ctDirect:      default:        UpdatesDirectCheck->Checked = true;        break;      case ctAuto:        UpdatesAutoCheck->Checked = true;        break;      case ctProxy:        UpdatesProxyCheck->Checked = true;        break;    }    UpdatesProxyHostEdit->Text = Updates.ProxyHost;    UpdatesProxyPortEdit->AsInteger = Updates.ProxyPort;    // presets    (*FCopyParamList) = *WinConfiguration->CopyParamList;    UpdateCopyParamListView();    CopyParamListView->ItemIndex = 0;    BOOLPROP(CopyParamAutoSelectNotice);    // interface    switch (CustomWinConfiguration->Interface)    {      case ifCommander:        CommanderInterfaceButton2->Checked = true;        break;      case ifExplorer:        ExplorerInterfaceButton2->Checked = true;        break;      default:        DebugFail();        break;    }    // security    UseMasterPasswordCheck->Checked = WinConfiguration->UseMasterPassword;    SessionRememberPasswordCheck->Checked = GUIConfiguration->SessionRememberPassword;    // network    RetrieveExternalIpAddressButton->Checked = Configuration->ExternalIpAddress.IsEmpty();    CustomExternalIpAddressButton->Checked = !RetrieveExternalIpAddressButton->Checked;    CustomExternalIpAddressEdit->Text = Configuration->ExternalIpAddress;    TryFtpWhenSshFailsCheck->Checked = Configuration->TryFtpWhenSshFails;    // logging    EnableLoggingCheck->Checked = Configuration->Logging;    LogProtocolCombo->ItemIndex = Configuration->LogProtocol + BelowNormalLogLevels;    LogFileNameEdit3->Text =      !Configuration->LogFileName.IsEmpty() ? Configuration->LogFileName : Configuration->DefaultLogFileName;    if (Configuration->LogFileAppend)    {      LogFileAppendButton->Checked = true;    }    else    {      LogFileOverwriteButton->Checked = true;    }    LogMaxSizeCheck->Checked = (Configuration->LogMaxSize > 0);    LogMaxSizeCombo->Text = SizeToStr((Configuration->LogMaxSize > 0) ? Configuration->LogMaxSize : (1 * 1024 * 1024));    LogMaxSizeCountCheck->Checked = (Configuration->LogMaxCount > 0);    LogMaxSizeCountEdit->AsInteger = ((Configuration->LogMaxCount > 0) ? Configuration->LogMaxCount : 5);    BOOLPROP(LogSensitive);    EnableActionsLoggingCheck->Checked = Configuration->LogActions;    ActionsLogFileNameEdit->Text = Configuration->ActionsLogFileName;    #undef BOOLPROP  }  __finally  {    FNoUpdate--;  }  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::SaveConfiguration(){  Configuration->BeginUpdate();  try  {    TGUICopyParamType CopyParam = GUIConfiguration->DefaultCopyParam;    #define BOOLPROP(PROP) WinConfiguration->PROP = PROP ## Check->Checked    BOOLPROP(DefaultDirIsHome);    BOOLPROP(PreservePanelState);    BOOLPROP(DeleteToRecycleBin);    BOOLPROP(DDWarnLackOfTempSpace);    BOOLPROP(ShowHiddenFiles);    BOOLPROP(RenameWholeName);    BOOLPROP(ShowInaccesibleDirectories);    BOOLPROP(CopyOnDoubleClickConfirmation);    BOOLPROP(ConfirmTransferring);    BOOLPROP(ConfirmOverwriting);    BOOLPROP(ConfirmResume);    BOOLPROP(ConfirmDeleting);    BOOLPROP(ConfirmRecycling);    BOOLPROP(ConfirmExitOnCompletion);    BOOLPROP(ConfirmCommandSession);    BOOLPROP(ContinueOnError);    BOOLPROP(SynchronizeSummary);    BOOLPROP(BeepOnFinish);    BOOLPROP(TemporaryDirectoryAppendSession);    BOOLPROP(TemporaryDirectoryAppendPath);    BOOLPROP(TemporaryDirectoryDeterministic);    BOOLPROP(TemporaryDirectoryCleanup);    BOOLPROP(ConfirmTemporaryDirectoryCleanup);    BOOLPROP(FullRowSelect);    BOOLPROP(NaturalOrderNumericalSorting);    WinConfiguration->ConfirmClosingSession = ConfirmClosingSessionCheck2->Checked;    WinConfiguration->DDTransferConfirmation =      CheckBoxAutoSwitchSave(DDTransferConfirmationCheck2);    GUIConfiguration->BeepOnFinishAfter =      static_cast<double>(BeepOnFinishAfterEdit->Value / SecsPerDay);    BOOLPROP(BalloonNotifications);    WinConfiguration->DDFakeFile = DDFakeFileEnabledButton->Checked;    if (DDSystemTemporaryDirectoryButton->Checked)    {      WinConfiguration->DDTemporaryDirectory = L"";    }    else    {      WinConfiguration->DDTemporaryDirectory = DDTemporaryDirectoryEdit->Text;    }    // Commander    TScpCommanderConfiguration ScpCommander = WinConfiguration->ScpCommander;    if (NortonLikeModeCombo->ItemIndex == 2)    {      ScpCommander.NortonLikeMode = nlOff;    }    else if (NortonLikeModeCombo->ItemIndex == 1)    {      ScpCommander.NortonLikeMode = nlKeyboard;    }    else    {      ScpCommander.NortonLikeMode = nlOn;    }    ScpCommander.SwappedPanels = SwappedPanelsCheck->Checked;    ScpCommander.TreeOnLeft = TreeOnLeftCheck->Checked;    ScpCommander.ExplorerKeyboardShortcuts =      (ExplorerKeyboardShortcutsCombo->ItemIndex != 0);    BOOLPROP(UseLocationProfiles);    ScpCommander.CompareByTime = CompareByTimeCheck->Checked;    ScpCommander.CompareBySize = CompareBySizeCheck->Checked;    // Local panel    ScpCommander.PreserveLocalDirectory = PreserveLocalDirectoryCheck->Checked;    ScpCommander.SystemContextMenu = SystemContextMenuCheck->Checked;    WinConfiguration->ScpCommander = ScpCommander;    // Explorer    TScpExplorerConfiguration ScpExplorer = WinConfiguration->ScpExplorer;    ScpExplorer.ShowFullAddress = ShowFullAddressCheck->Checked;    WinConfiguration->ScpExplorer = ScpExplorer;    Configuration->RandomSeedFile = RandomSeedFileEdit->Text;    // editor    WinConfiguration->Editor.WordWrap = EditorWordWrapCheck->Checked;    WinConfiguration->Editor.TabSize = EditorTabSizeEdit->AsInteger;    switch (EditorEncodingCombo->ItemIndex)    {      case 1:        WinConfiguration->Editor.Encoding = CP_UTF8;        break;      default:        WinConfiguration->Editor.Encoding = CP_ACP;        break;    }    TWinConfiguration::StoreFont(FEditorFont.get(), WinConfiguration->Editor.Font);    WinConfiguration->Editor.FontColor = FEditorFont->Color;    WinConfiguration->Editor.BackgroundColor = FEditorBackgroundColor;    WinConfiguration->EditorList = FEditorList;    // overwrites only TCopyParamType fields    CopyParam = FCopyParams;    if (ResumeOnButton->Checked) CopyParam.ResumeSupport = rsOn;    if (ResumeSmartButton->Checked) CopyParam.ResumeSupport = rsSmart;    if (ResumeOffButton->Checked) CopyParam.ResumeSupport = rsOff;    CopyParam.ResumeThreshold = ResumeThresholdEdit->AsInteger * 1024;    Configuration->SessionReopenAuto =      (SessionReopenAutoCheck->Checked ? (SessionReopenAutoEdit->AsInteger * MSecsPerSec) : 0);    GUIConfiguration->SessionReopenAutoIdle =      (SessionReopenAutoIdleCheck->Checked ? (SessionReopenAutoIdleEdit->AsInteger * MSecsPerSec) : 0);    Configuration->SessionReopenAutoStall =      (SessionReopenAutoStallCheck->Checked ? (SessionReopenAutoStallEdit->AsInteger * MSecsPerSec) : 0);    Configuration->SessionReopenTimeout = (SessionReopenTimeoutEdit->AsInteger * MSecsPerSec);    WinConfiguration->CustomCommandList = FCustomCommandList;    WinConfiguration->ExtensionList = FExtensionList;    WinConfiguration->CustomCommandOptions = FCustomCommandOptions.get();    GUIConfiguration->PuttyPath = PuttyPathEdit->Text;    GUIConfiguration->PuttyPassword = PuttyPasswordCheck2->Checked;    WinConfiguration->AutoOpenInPutty = AutoOpenInPuttyCheck->Checked;    WinConfiguration->TelnetForFtpInPutty = TelnetForFtpInPuttyCheck->Checked;    // do not overwrite custom keys    if (PuttyRegistryStorageKeyEdit->ItemIndex >= 0)    {      GUIConfiguration->PuttyRegistryStorageKey = PuttyRegistryStorageKeyEdit->Text;    }    // Queue    GUIConfiguration->QueueTransfersLimit = QueueTransferLimitEdit->AsInteger;    WinConfiguration->EnableQueueByDefault = EnableQueueByDefaultCheck->Checked;    GUIConfiguration->QueueAutoPopup = QueueAutoPopupCheck->Checked;    CopyParam.Queue = QueueCheck->Checked;    CopyParam.QueueParallel = QueueParallelCheck->Checked;    CopyParam.QueueNoConfirmation = QueueNoConfirmationCheck->Checked;    GUIConfiguration->QueueKeepDoneItems = (QueueKeepDoneItemsForCombo->ItemIndex != 0);    switch (QueueKeepDoneItemsForCombo->ItemIndex)    {      case 0:        GUIConfiguration->QueueKeepDoneItemsFor = 0;        break;      case 1:        GUIConfiguration->QueueKeepDoneItemsFor = 15;        break;      case 2:        GUIConfiguration->QueueKeepDoneItemsFor = 60;        break;      case 3:        GUIConfiguration->QueueKeepDoneItemsFor = 15 * 60;        break;      case 4:        GUIConfiguration->QueueKeepDoneItemsFor = 60 * 60;        break;      default:        GUIConfiguration->QueueKeepDoneItemsFor = -1;        break;    }    if (QueueViewShowButton->Checked)    {      WinConfiguration->QueueView.Show = qvShow;    }    else if (QueueViewHideWhenEmptyButton->Checked)    {      WinConfiguration->QueueView.Show = qvHideWhenEmpty;    }    else    {      WinConfiguration->QueueView.Show = qvHide;    }    GUIConfiguration->DefaultCopyParam = CopyParam;    // window    WinConfiguration->AutoSaveWorkspace =      !AutoWorkspaceCombo->Text.IsEmpty() &&      AutoSaveWorkspaceCheck->Checked;    if (!AutoWorkspaceCombo->Text.IsEmpty())    {      WinConfiguration->AutoWorkspace = AutoWorkspaceCombo->Text;    }    WinConfiguration->AutoSaveWorkspacePasswords = AutoSaveWorkspacePasswordsCheck->Checked;    if (PathInCaptionFullButton->Checked)    {       WinConfiguration->PathInCaption = picFull;    }    else if (PathInCaptionShortButton->Checked)    {      WinConfiguration->PathInCaption = picShort;    }    else    {      WinConfiguration->PathInCaption = picNone;    }    BOOLPROP(MinimizeToTray);    BOOLPROP(ExternalSessionInExistingInstance);    BOOLPROP(KeepOpenWhenNoSession);    BOOLPROP(ShowTips);    // panels    WinConfiguration->DoubleClickAction = (TDoubleClickAction)DoubleClickActionCombo->ItemIndex;    BOOLPROP(AutoReadDirectoryAfterOp);    BOOLPROP(RefreshRemotePanel);    WinConfiguration->RefreshRemotePanelInterval =      static_cast<double>(RefreshRemotePanelIntervalEdit->Value / SecsPerDay);    switch (FormatSizeBytesCombo->ItemIndex)    {      case 0:        WinConfiguration->FormatSizeBytes = fbNone;        break;      case 1:        WinConfiguration->FormatSizeBytes = fbKilobytes;        break;      case 2:        WinConfiguration->FormatSizeBytes = fbShort;        break;      default:        DebugFail();    }    TFontConfiguration PanelFontConfiguration;    if (PanelFontCheck->Checked)    {      TWinConfiguration::StoreFont(FPanelFont.get(), PanelFontConfiguration);    }    WinConfiguration->PanelFont = PanelFontConfiguration;    // updates    WinConfiguration->Updates = SaveUpdates();    Configuration->CollectUsage = CollectUsageCheck->Checked;    // presets    WinConfiguration->CopyParamList = FCopyParamList;    BOOLPROP(CopyParamAutoSelectNotice);    // interface    if (GetInterface() != CustomWinConfiguration->Interface)    {      Configuration->Usage->Inc(L"InterfaceChanges");    }    CustomWinConfiguration->Interface = GetInterface();    // network    Configuration->ExternalIpAddress =      (CustomExternalIpAddressButton->Checked ? CustomExternalIpAddressEdit->Text : UnicodeString());    Configuration->TryFtpWhenSshFails = TryFtpWhenSshFailsCheck->Checked;    // security    GUIConfiguration->SessionRememberPassword = SessionRememberPasswordCheck->Checked;    // logging    Configuration->Logging = EnableLoggingCheck->Checked && !LogFileNameEdit3->Text.IsEmpty();    Configuration->LogProtocol = LogProtocolCombo->ItemIndex - BelowNormalLogLevels;    Configuration->LogFileName = LogFileNameEdit3->Text;    Configuration->LogFileAppend = LogFileAppendButton->Checked;    __int64 LogMaxSize;    if (LogMaxSizeCheck->Checked && DebugAlwaysTrue(TryStrToSize(LogMaxSizeCombo->Text, LogMaxSize)))    {      Configuration->LogMaxSize = LogMaxSize;    }    else    {      Configuration->LogMaxSize = 0;    }    Configuration->LogMaxCount = (LogMaxSizeCountCheck->Checked ? LogMaxSizeCountEdit->AsInteger : 0);    BOOLPROP(LogSensitive);    Configuration->LogActions = EnableActionsLoggingCheck->Checked;    Configuration->ActionsLogFileName = ActionsLogFileNameEdit->Text;    // languages    if (LanguagesView->ItemFocused != NULL)    {      TLocaleInfo * LocaleInfo = static_cast<TLocaleInfo *>(LanguagesView->ItemFocused->Data);      LCID Locale;      // Do not change the locale settings, unless changed explicitly by user      // to allow an automatic upgrade to new translation once the UI language translation      // becomes available      if (LocaleInfo->Locale == WinConfiguration->DefaultLocale)      {        Locale = NULL;      }      else      {        Locale = LocaleInfo->Locale;      }      GUIConfiguration->Locale = Locale;    }    #undef BOOLPROP  }  __finally  {    Configuration->EndUpdate();  }  bool MoveStorage = true;  TStorage Storage;  if (RegistryStorageButton->Checked)  {    Storage = stRegistry;  }  else if (AutomaticIniFileStorageButton->Checked)  {    Storage = stIniFile;  }  else if (CustomIniFileStorageButton->Checked)  {    Storage = stIniFile;  }  else  {    MoveStorage = false;  }  if (MoveStorage)  {    Configuration->MoveStorage(Storage, GetCustomIniFileStorageName());  }}//---------------------------------------------------------------------------TUpdatesConfiguration __fastcall TPreferencesDialog::SaveUpdates(){  TUpdatesConfiguration Updates = WinConfiguration->Updates;  if (UpdatesPeriodCombo->ItemIndex == 0)  {    Updates.Period = 0;  }  else if (UpdatesPeriodCombo->ItemIndex == 1)  {    Updates.Period = 1;  }  else if (UpdatesPeriodCombo->ItemIndex == 2)  {    Updates.Period = 7;  }  else  {    Updates.Period = 30;  }  Updates.ShowOnStartup = UpdatesShowOnStartup->Checked;  Updates.AuthenticationEmail = UpdatesAuthenticationEmailEdit->Text;  Updates.BetaVersions = ComboAutoSwitchSave(UpdatesBetaVersionsCombo);  if (UpdatesDirectCheck->Checked)  {    Updates.ConnectionType = ctDirect;  }  else if (UpdatesAutoCheck->Checked)  {    Updates.ConnectionType = ctAuto;  }  else if (UpdatesProxyCheck->Checked)  {    if (!UpdatesProxyHostEdit->Text.IsEmpty())    {      Updates.ConnectionType = ctProxy;    }    else    {      Updates.ConnectionType = ctDirect;    }  }  Updates.ProxyHost = UpdatesProxyHostEdit->Text;  Updates.ProxyPort = UpdatesProxyPortEdit->AsInteger;  return Updates;}//---------------------------------------------------------------------------TInterface __fastcall TPreferencesDialog::GetInterface(){  return CommanderInterfaceButton2->Checked ? ifCommander : ifExplorer;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::FormShow(TObject * /*Sender*/){  // Load only now, particularly so that we have handles allocated already.  // If called before ShowModal, we end up recreating the handles,  // what causes flicker of the currently focused window  LoadConfiguration();  InstallPathWordBreakProc(RandomSeedFileEdit);  InstallPathWordBreakProc(DDTemporaryDirectoryEdit);  InstallPathWordBreakProc(PuttyPathEdit);  InstallPathWordBreakProc(LogFileNameEdit3);  InstallPathWordBreakProc(ActionsLogFileNameEdit);  ListView_SetExtendedListViewStyle(CustomCommandsView->Handle, (ListView_GetExtendedListViewStyle(CustomCommandsView->Handle) & ~LVS_EX_INFOTIP));  PrepareNavigationTree(NavigationTree);  switch (FPreferencesMode) {    case pmEditor: PageControl->ActivePage = EditorSheet; break;    case pmCustomCommands: PageControl->ActivePage = CustomCommandsSheet; break;    case pmQueue: PageControl->ActivePage = QueueSheet; break;    case pmLogging: PageControl->ActivePage = LogSheet; break;    case pmUpdates: PageControl->ActivePage = UpdatesSheet; break;    case pmPresets: PageControl->ActivePage = CopyParamListSheet; break;    case pmEditors: PageControl->ActivePage = EditorSheet; break;    case pmCommander: PageControl->ActivePage = CommanderSheet; break;    case pmEditorInternal: PageControl->ActivePage = EditorInternalSheet; break;    default: PageControl->ActivePage = PreferencesSheet; break;  }  PageControlChange(NULL);  ActiveControl = NavigationTree;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ControlChange(TObject * /*Sender*/){  UpdateControls();}//---------------------------------------------------------------------------UnicodeString __fastcall TPreferencesDialog::TabSample(UnicodeString Values){  UnicodeString Result;  for (int Index = 1; Index <= Values.Length(); Index++)  {    if (Index > 1)    {      Result += L' ';      if (EditorTabSizeEdit->AsInteger > 2)      {        Result += UnicodeString::StringOfChar(L' ', EditorTabSizeEdit->AsInteger - 2);      }    }    Result += Values[Index];  }  return Result;}//---------------------------------------------------------------------------TCustomCommandList * __fastcall TPreferencesDialog::GetCommandList(int Index){  if (Index < FCustomCommandList->Count)  {    return FCustomCommandList;  }  else  {    return FExtensionList;  }}//---------------------------------------------------------------------------int __fastcall TPreferencesDialog::GetCommandIndex(int Index){  if (Index >= FCustomCommandList->Count)  {    Index -= FCustomCommandList->Count;  }  return Index;}//---------------------------------------------------------------------------int __fastcall TPreferencesDialog::GetListCommandIndex(TCustomCommandList * List){  int Index;  if (GetCommandList(CustomCommandsView->ItemIndex) == List)  {    Index = GetCommandIndex(CustomCommandsView->ItemIndex);  }  else  {    Index = -1;  }  return Index;}//---------------------------------------------------------------------------int __fastcall TPreferencesDialog::GetCommandListIndex(TCustomCommandList * List, int Index){  if (List == FExtensionList)  {    Index += FCustomCommandList->Count;  }  return Index;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdateControls(){  if (FNoUpdate == 0)  {    EnableControl(DDTransferConfirmationCheck2, ConfirmTransferringCheck->Checked);    EnableControl(BeepOnFinishAfterEdit, BeepOnFinishCheck->Checked);    EnableControl(BeepOnFinishAfterText, BeepOnFinishCheck->Checked);    EnableControl(ResumeThresholdEdit, ResumeSmartButton->Checked);    EnableControl(ResumeThresholdUnitLabel2, ResumeThresholdEdit->Enabled);    EnableControl(SessionReopenAutoEdit, SessionReopenAutoCheck->Checked);    EnableControl(SessionReopenAutoLabel, SessionReopenAutoEdit->Enabled);    EnableControl(SessionReopenAutoSecLabel, SessionReopenAutoEdit->Enabled);    EnableControl(SessionReopenAutoIdleEdit, SessionReopenAutoIdleCheck->Checked);    EnableControl(SessionReopenAutoIdleLabel, SessionReopenAutoIdleEdit->Enabled);    EnableControl(SessionReopenAutoIdleSecLabel, SessionReopenAutoIdleEdit->Enabled);    EnableControl(SessionReopenAutoStallEdit, SessionReopenAutoStallCheck->Checked);    EnableControl(SessionReopenAutoStallLabel, SessionReopenAutoStallEdit->Enabled);    EnableControl(SessionReopenAutoStallSecLabel, SessionReopenAutoStallEdit->Enabled);    EnableControl(SessionReopenTimeoutEdit,      SessionReopenAutoEdit->Enabled || SessionReopenAutoStallCheck->Checked);    EnableControl(SessionReopenTimeoutLabel, SessionReopenTimeoutEdit->Enabled);    EnableControl(SessionReopenTimeoutSecLabel,SessionReopenTimeoutEdit->Enabled);    EnableControl(CopyOnDoubleClickConfirmationCheck,      (DoubleClickActionCombo->ItemIndex == 1) && ConfirmTransferringCheck->Checked);    TFont * ActualPanelFont = PanelFontCheck->Checked ? FPanelFont.get() : Screen->IconFont;    std::unique_ptr<TFont> PanelFont(new TFont());    PanelFont->Assign(ActualPanelFont);    if (PanelFontCheck->Checked)    {      PanelFont->Height = ScaleByPixelsPerInchFromSystem(PanelFont->Height, this);    }    else    {      PanelFont->Height = ScaleByPixelsPerInchFromSystem(PanelFont->Height, this);    }    UnicodeString PanelFontLabelText;    PanelFontLabelText = FMTLOAD(EDITOR_FONT_FMT,      (ActualPanelFont->Name, ActualPanelFont->Size));    PanelFontLabel->Caption = PanelFontLabelText;    if (!SameFont(PanelFontLabel->Font, PanelFont.get()))    {      PanelFontLabel->Font = PanelFont.get();    }    EnableControl(RefreshRemotePanelIntervalEdit, RefreshRemotePanelCheck->Checked);    EnableControl(RefreshRemoteDirectoryUnitLabel, RefreshRemotePanelCheck->Checked);    UnicodeString EditorFontLabelText;    EditorFontLabelText = FMTLOAD(EDITOR_FONT_FMT,      (FEditorFont->Name, FEditorFont->Size)) + L"\n\n";    EditorFontLabelText += TabSample(L"ABCD") + L"\n";    EditorFontLabelText += TabSample(L"1234");    EditorFontLabel->Caption = EditorFontLabelText;    std::unique_ptr<TFont> EditorFont(new TFont());    EditorFont->Assign(FEditorFont.get());    EditorFont->Color = GetWindowTextColor(FEditorFont->Color);    EditorFont->Size = ScaleByPixelsPerInchFromSystem(FEditorFont->Size, this);    if (!SameFont(EditorFontLabel->Font, EditorFont.get()) ||        (EditorFontLabel->Font->Color != EditorFont->Color))    {      EditorFontLabel->Font = EditorFont.get();    }    EditorFontLabel->Color = GetWindowColor(FEditorBackgroundColor);    TCustomCommandList * CommandList = GetCommandList(CustomCommandsView->ItemIndex);    int CommandIndex = GetCommandIndex(CustomCommandsView->ItemIndex);    bool CommandSelected = (CustomCommandsView->Selected != NULL);    bool CustomCommandSelected = CommandSelected && (CommandList == FCustomCommandList);    bool ExtensionSelected = CommandSelected && (CommandList == FExtensionList);    EnableControl(EditCommandButton, CustomCommandSelected);    EditCommandButton->Visible = !ExtensionSelected;    EnableControl(ConfigureCommandButton, ExtensionSelected);    ConfigureCommandButton->Visible = ExtensionSelected;    EnableControl(RemoveCommandButton, CommandSelected);    EnableControl(UpCommandButton, CommandSelected && (CommandIndex > 0));    EnableControl(DownCommandButton, CommandSelected && (CommandIndex < CommandList->Count - 1));    bool CopyParamSelected = (CopyParamListView->Selected != NULL);    EnableControl(EditCopyParamButton, CopyParamSelected);    EnableControl(DuplicateCopyParamButton, CopyParamSelected);    EnableControl(RemoveCopyParamButton,      CopyParamSelected && (CopyParamListView->ItemIndex >= 1));    EnableControl(UpCopyParamButton,      CopyParamSelected && (CopyParamListView->ItemIndex > 1));    EnableControl(DownCopyParamButton,      CopyParamSelected &&      (CopyParamListView->ItemIndex >= 1) &&      (CopyParamListView->ItemIndex < CopyParamListView->Items->Count - 1));    EnableControl(CopyParamAutoSelectNoticeCheck, FCopyParamList->AnyRule);    UnicodeString InfoStr;    if (CopyParamSelected)    {      const TCopyParamType * SelectedCopyParam = GetCopyParam(CopyParamListView->ItemIndex);      InfoStr = SelectedCopyParam->GetInfoStr(L"; ", 0);      if (CopyParamListView->ItemIndex >= 1)      {        const TCopyParamRule * Rule = FCopyParamList->Rules[CopyParamListView->ItemIndex - 1];        if (Rule != NULL)        {          InfoStr += L"\n" + FORMAT(ReplaceStr(LoadStr(COPY_PARAM_RULE), L"\n", L" "), (Rule->GetInfoStr(L"; ")));        }      }    }    SetLabelHintPopup(CopyParamLabel, InfoStr);    if (!WinConfiguration->DDExtInstalled)    {      DragExtStatusLabel->Caption = LoadStr(PREFERENCES_DRAGEXT_NOT_INSTALLED);      DragExtStatusLabel->Enabled = false;      DragExtStatusLabel->Font->Color = clWindowText;    }    else if (!WinConfiguration->IsDDExtRunning())    {      DragExtStatusLabel->Caption = LoadStr(PREFERENCES_DRAGEXT_NOT_RUNNING);      DragExtStatusLabel->Enabled = true;      DragExtStatusLabel->Font->Color = clGrayText;    }    else    {      DragExtStatusLabel->Caption = LoadStr(PREFERENCES_DRAGEXT_RUNNING);      DragExtStatusLabel->Enabled = true;      DragExtStatusLabel->Font->Color = clWindowText;    }    EnableControl(DDFakeFileDisabledPanel, DDFakeFileDisabledButton->Checked);    EnableControl(DDTemporaryDirectoryEdit, DDCustomTemporaryDirectoryButton->Enabled &&      DDCustomTemporaryDirectoryButton->Checked);    EnableControl(ConfirmTemporaryDirectoryCleanupCheck,      TemporaryDirectoryCleanupCheck->Checked);    // allow only when some of the known storages is selected,    // and particularly do not allow switching storage, when we start with stNul,    // as that would destroy the stored configuration    EnableControl(StorageGroup,      RegistryStorageButton->Checked || AutomaticIniFileStorageButton->Checked || CustomIniFileStorageButton->Checked);    AutomaticIniFileStorageLabel->UpdateStatus();    EnableControl(CustomIniFileStorageEdit, CustomIniFileStorageButton->Checked);    EditorFontLabel->WordWrap = EditorWordWrapCheck->Checked;    bool EditorSelected = (EditorListView3->Selected != NULL);    EnableControl(EditEditorButton, EditorSelected);    EnableControl(RemoveEditorButton, EditorSelected);    EnableControl(UpEditorButton, EditorSelected &&      (EditorListView3->ItemIndex > 0));    EnableControl(DownEditorButton, EditorSelected &&      (EditorListView3->ItemIndex < EditorListView3->Items->Count - 1));    // updates    EnableControl(UpdatesAuthenticationEmailEdit, FAutomaticUpdatesPossible);    EnableControl(UpdatesAuthenticationEmailLabel, UpdatesAuthenticationEmailEdit->Enabled);    EnableControl(UpdatesShowOnStartup, !IsUWP());    EnableControl(UsageViewButton, CollectUsageCheck->Checked);    EnableControl(UpdatesProxyHostEdit, UpdatesProxyCheck->Checked);    EnableControl(UpdatesProxyHostLabel, UpdatesProxyHostEdit->Enabled);    EnableControl(UpdatesProxyPortEdit, UpdatesProxyCheck->Checked);    EnableControl(UpdatesProxyPortLabel, UpdatesProxyPortEdit->Enabled);    bool IsSiteCommand = false;    bool IsPasswordCommand = false;    try    {      TRemoteCustomCommand RemoteCustomCommand;      TInteractiveCustomCommand InteractiveCustomCommand(&RemoteCustomCommand);      UnicodeString PuttyPath = PuttyPathEdit->Text;      PuttyPath = InteractiveCustomCommand.Complete(PuttyPath, false);      IsSiteCommand = RemoteCustomCommand.IsSiteCommand(PuttyPath);      IsPasswordCommand = RemoteCustomCommand.IsPasswordCommand(PuttyPath);    }    catch (...)    {      // noop    }    bool AnyPuttyPath = !PuttyPathEdit->Text.IsEmpty();    EnableControl(PuttyPasswordCheck2, AnyPuttyPath && !IsPasswordCommand);    EnableControl(AutoOpenInPuttyCheck, AnyPuttyPath);    EnableControl(TelnetForFtpInPuttyCheck,      AnyPuttyPath && !IsSiteCommand);    EnableControl(PuttyRegistryStorageKeyEdit,      AnyPuttyPath && !IsSiteCommand && !IsUWP());    EnableControl(PuttyRegistryStorageKeyLabel, PuttyRegistryStorageKeyEdit->Enabled);    EnableControl(SetMasterPasswordButton, WinConfiguration->UseMasterPassword);    // network    EnableControl(CustomExternalIpAddressEdit, CustomExternalIpAddressButton->Checked);    // window    EnableControl(AutoWorkspaceCombo, AutoSaveWorkspaceCheck->Checked);    EnableControl(AutoWorkspaceLabel, AutoWorkspaceCombo->Enabled);    EnableControl(AutoSaveWorkspacePasswordsCheck,      !Configuration->DisablePasswordStoring &&      AutoWorkspaceCombo->Enabled);    // integration    EnableControl(ShellIconsGroup, !IsUWP());    // There's no quick launch in Windows 7    EnableControl(QuickLaunchIconButton, ShellIconsGroup->Enabled && !IsWin7());    MakeDefaultHandlerItem->Visible = IsWinVista();    // languages    LanguageChangeLabel->Visible =      DebugAlwaysTrue(!GUIConfiguration->CanApplyLocaleImmediately) &&      (LanguagesView->ItemFocused != NULL) &&      (static_cast<TLocaleInfo *>(LanguagesView->ItemFocused->Data)->Locale != GUIConfiguration->AppliedLocale);    // logging    EnableControl(LogProtocolCombo, EnableLoggingCheck->Checked);    EnableControl(LogFileNameEdit3, LogProtocolCombo->Enabled);    EnableControl(LogFileNameHintText, LogFileNameEdit3->Enabled);    EnableControl(LogFileAppendButton, LogFileNameEdit3->Enabled);    EnableControl(LogFileOverwriteButton, LogFileNameEdit3->Enabled);    EnableControl(LogMaxSizeCheck, LogFileNameEdit3->Enabled);    EnableControl(LogMaxSizeCombo, LogMaxSizeCheck->Enabled && LogMaxSizeCheck->Checked);    EnableControl(LogMaxSizeCountCheck, LogMaxSizeCombo->Enabled);    EnableControl(LogMaxSizeCountEdit, LogMaxSizeCountCheck->Enabled && LogMaxSizeCountCheck->Checked);    EnableControl(LogMaxSizeCountFilesLabel, LogMaxSizeCountEdit->Enabled);    EnableControl(LogSensitiveCheck, LogProtocolCombo->Enabled);    EnableControl(ActionsLogFileNameEdit, EnableActionsLoggingCheck->Checked);    EnableControl(ActionsLogFileNameHintText, ActionsLogFileNameEdit->Enabled);    // interface    InterfaceChangeLabel->Visible =      !CustomWinConfiguration->CanApplyInterfaceImmediately &&      (GetInterface() != CustomWinConfiguration->AppliedInterface);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorFontButtonClick(TObject * /*Sender*/){  if (FontDialog(FEditorFont.get()))  {    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorFontColorChange(TColor Color){  FEditorFont->Color = Color;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorFontColorButtonClick(TObject * /*Sender*/){  // WORKAROUND: Compiler keeps crashing randomly (but frequently) with  // "internal error" when passing menu directly to unique_ptr.  // Splitting it to two statements seems to help.  // The same hack exists in TSiteAdvancedDialog::ColorButtonClick and TOpenLocalPathHandler::Open  TPopupMenu * Menu = CreateColorPopupMenu(FEditorFont->Color, EditorFontColorChange);  // Popup menu has to survive the popup as TBX calls click handler asynchronously (post).  FColorPopupMenu.reset(Menu);  MenuPopup(Menu, EditorFontColorButton);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorBackgroundColorChange(TColor Color){  FEditorBackgroundColor = Color;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorBackgroundColorButtonClick(TObject * /*Sender*/){  // See comment in EditorFontColorButtonClick.  // We are using session color (contrary to editor text color) for background  // for a consistency with color selection menu on editor toolbar.  TTBXPopupMenu * PopupMenu = new TTBXPopupMenu(Application);  FColorPopupMenu.reset(PopupMenu);  CreateEditorBackgroundColorMenu(PopupMenu->Items, FEditorBackgroundColor, EditorBackgroundColorChange);  MenuPopup(FColorPopupMenu.get(), EditorBackgroundColorButton);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PanelFontButtonClick(TObject * /*Sender*/){  if (FontDialog(FPanelFont.get()))  {    PanelFontCheck->Checked = true;    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::FormCloseQuery(TObject * /*Sender*/,  bool & /*CanClose*/){  if (ModalResult == DefaultResult(this))  {    ExitActiveControl(this);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::IconButtonClick(TObject *Sender){  UnicodeString IconName, Params;  int SpecialFolder;  if (Sender == DesktopIconButton)  {    IconName = AppName;    int Result =      MessageDialog(LoadStr(CREATE_DESKTOP_ICON2), qtConfirmation,        qaYes | qaNo | qaCancel, HELP_CREATE_ICON);    switch (Result)    {      case qaYes:        SpecialFolder = CSIDL_COMMON_DESKTOPDIRECTORY;        break;      case qaNo:        SpecialFolder = CSIDL_DESKTOPDIRECTORY;        break;      default:        Abort();        break;    }  }  else  {    if (MessageDialog(MainInstructions(LoadStr(CONFIRM_CREATE_ICON)),          qtConfirmation, qaYes | qaNo, HELP_CREATE_ICON) == qaYes)    {      if (Sender == SendToHookButton)      {        IconName = FMTLOAD(SENDTO_HOOK_NAME2, (AppName));        SpecialFolder = CSIDL_SENDTO;        Params = TProgramParams::FormatSwitch(UPLOAD_SWITCH);      }      else if (Sender == QuickLaunchIconButton)      {        IconName = L"Microsoft\\Internet Explorer\\Quick Launch\\" +          AppName;        SpecialFolder = CSIDL_APPDATA;      }    }    else    {      Abort();    }  }  TInstantOperationVisualizer Visualizer;  CreateDesktopShortCut(IconName,    Application->ExeName, Params, L"", SpecialFolder);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewData(TObject * /*Sender*/,      TListItem * Item){  // WORKAROUND We get here on Wine after destructor is called  if ((FCustomCommandList != NULL) && (FExtensionList != NULL))  {    int Index = Item->Index;    const TCustomCommandType * Command = GetCommandList(Index)->Commands[GetCommandIndex(Index)];    UnicodeString Caption = StripHotkey(Command->Name);    Item->Caption = Caption;    DebugAssert(!Item->SubItems->Count);    Item->SubItems->Add(Command->Command);    Item->SubItems->Add(LoadStr(      FLAGSET(Command->Params, ccLocal) ? CUSTOM_COMMAND_LOCAL : CUSTOM_COMMAND_REMOTE));  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ListViewSelectItem(  TObject * /*Sender*/, TListItem * /*Item*/, bool /*Selected*/){  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdateCustomCommandsView(){  CustomCommandsView->Items->Count = FCustomCommandList->Count + FExtensionList->Count;  AutoSizeListColumnsWidth(CustomCommandsView, 1);  CustomCommandsView->Invalidate();  // particularly after command is edited/configured, make sure the hint is updated,  // even if we manage to display a hint for the same command as before the change  FCustomCommandsHintItem = NULL;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewKeyDown(      TObject * /*Sender*/, WORD & Key, TShiftState /*Shift*/){  if (RemoveCommandButton->Enabled && (Key == VK_DELETE))  {    RemoveCommandButtonClick(NULL);  }  if (AddCommandButton->Enabled && (Key == VK_INSERT))  {    AddEditCommand(false);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewDblClick(  TObject * /*Sender*/){  if (EditCommandButton->Enabled)  {    AddEditCommand(true);  }  else if (ConfigureCommandButton->Enabled)  {    ConfigureCommand();  }}//---------------------------------------------------------------------------static int __fastcall AddCommandToList(TCustomCommandList * List, int Index, TCustomCommandType * Command){  if (Index >= 0)  {    List->Insert(Index, Command);  }  else  {    List->Add(Command);    Index = List->Count - 1;  }  return Index;}//---------------------------------------------------------------------------TShortCuts __fastcall TPreferencesDialog::GetShortCuts(){  TShortCuts ShortCuts;  if (WinConfiguration->SharedBookmarks != NULL)  {    WinConfiguration->SharedBookmarks->ShortCuts(ShortCuts);  }  FCustomCommandList->ShortCuts(ShortCuts);  FExtensionList->ShortCuts(ShortCuts);  return ShortCuts;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddEditCommand(bool Edit){  TCustomCommandType Command;  if (Edit)  {    int Index = CustomCommandsView->ItemIndex;    DebugAssert(GetCommandList(Index) == FCustomCommandList);    Command = *FCustomCommandList->Commands[GetCommandIndex(Index)];  }  TShortCuts ShortCuts = GetShortCuts();  if (DoCustomCommandDialog(Command, FCustomCommandList,        (Edit ? ccmEdit : ccmAdd), 0, NULL, &ShortCuts))  {    int Index = GetListCommandIndex(FCustomCommandList);    TCustomCommandType * ACommand = new TCustomCommandType(Command);    if (Edit)    {      DebugAssert(Index < FCustomCommandList->Count);      FCustomCommandList->Change(Index, ACommand);    }    else    {      Index = AddCommandToList(FCustomCommandList, Index, ACommand);    }    UpdateCustomCommandsView();    CustomCommandsView->ItemIndex = GetCommandListIndex(FCustomCommandList, Index);    CustomCommandsView->ItemFocused->MakeVisible(false);    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::RemoveCommandButtonClick(      TObject * /*Sender*/){  TCustomCommandList * List = GetCommandList(CustomCommandsView->ItemIndex);  int Index = GetCommandIndex(CustomCommandsView->ItemIndex);  if (List == FExtensionList)  {    const TCustomCommandType * CustomComand = List->Commands[Index];    // If the extension was added in this "preferences session", remove the file    int PathIndex = FAddedExtensions->IndexOf(CustomComand->FileName);    if (PathIndex >= 0)    {      FAddedExtensions->Delete(PathIndex);      DeleteFile(ApiPath(CustomComand->FileName));    }  }  List->Delete(Index);  UpdateCustomCommandsView();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandMove(int Source, int Dest){  TCustomCommandList * List = GetCommandList(CustomCommandsView->ItemIndex);  int SourceIndex = GetCommandIndex(Source);  int DestIndex = GetCommandIndex(Dest);  List->Move(SourceIndex, DestIndex);  // workaround for bug in VCL  CustomCommandsView->ItemIndex = -1;  CustomCommandsView->ItemFocused = CustomCommandsView->Selected;  CustomCommandsView->ItemIndex = Dest;  UpdateCustomCommandsView();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpDownCommandButtonClick(TObject * Sender){  CustomCommandMove(CustomCommandsView->ItemIndex,    CustomCommandsView->ItemIndex + (Sender == UpCommandButton ? -1 : 1));}//---------------------------------------------------------------------------TListViewScrollOnDragOver * __fastcall TPreferencesDialog::ScrollOnDragOver(TObject * ListView){  if (ListView == CopyParamListView)  {    return FCopyParamScrollOnDragOver;  }  else if (ListView == CustomCommandsView)  {    return FCustomCommandsScrollOnDragOver;  }  else if (ListView == EditorListView3)  {    return FEditorScrollOnDragOver;  }  else  {    DebugFail();    return NULL;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ListViewStartDrag(      TObject * Sender, TDragObject *& /*DragObject*/){  FListViewDragSource = dynamic_cast<TListView*>(Sender)->ItemIndex;  FListViewDragDest = -1;  ScrollOnDragOver(Sender)->StartDrag();}//---------------------------------------------------------------------------static int __fastcall PointToListViewIndex(TObject * Sender, int X, int Y){  TListItem * Item = dynamic_cast<TListView*>(Sender)->GetItemAt(X, Y);  return Item ? Item->Index : -1;}//---------------------------------------------------------------------------bool __fastcall TPreferencesDialog::AllowListViewDrag(TObject * Sender, int X, int Y){  FListViewDragDest = PointToListViewIndex(Sender, X, Y);  return (FListViewDragDest >= 0) && (FListViewDragDest != FListViewDragSource);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewDragDrop(      TObject * Sender, TObject * Source, int X, int Y){  if (Source == CustomCommandsView)  {    if (AllowListViewDrag(Sender, X, Y) &&        (GetCommandList(FListViewDragSource) == GetCommandList(FListViewDragDest)))    {      CustomCommandMove(FListViewDragSource, FListViewDragDest);    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewDragOver(  TObject *Sender, TObject *Source, int X, int Y, TDragState State, bool & Accept){  ListViewDragOver(Sender, Source, X, Y, State, Accept);  if (Source == Sender)  {    int Dest = PointToListViewIndex(Sender, X, Y);    if (GetCommandList(FListViewDragSource) != GetCommandList(Dest))    {      Accept = false;    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ListViewDragOver(  TObject * Sender, TObject * Source, int X, int Y,  TDragState /*State*/, bool & Accept){  if (Source == Sender)  {    // cannot use AllowListViewDrag(X, Y) because of bug in VCL    // (when dropped on item itself, when it was dragged over another item before,    // that another item remains highlighted forever)    Accept = true;    ScrollOnDragOver(Source)->DragOver(TPoint(X, Y));  }}//---------------------------------------------------------------------------const TCopyParamType * TPreferencesDialog::GetCopyParam(int Index){  if (Index == 0)  {    return &FCopyParams;  }  else  {    return FCopyParamList->CopyParams[Index - 1];  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamMove(int Source, int Dest){  if (Source >= 1 && Source < (1 + FCopyParamList->Count) &&      Dest >= 0 && Dest < (1 + FCopyParamList->Count))  {    if (Dest == 0)    {      Dest = 1;    }    FCopyParamList->Move(Source - 1, Dest - 1);    // workaround for bug in VCL    CopyParamListView->ItemIndex = -1;    CopyParamListView->ItemFocused = CopyParamListView->Selected;    CopyParamListView->ItemIndex = Dest;    UpdateCopyParamListView();    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamListViewDragDrop(  TObject * Sender, TObject * Source, int X, int Y){  if (Source == CopyParamListView)  {    if (AllowListViewDrag(Sender, X, Y))    {      CopyParamMove(FListViewDragSource, FListViewDragDest);    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpDownCopyParamButtonClick(TObject * Sender){  CopyParamMove(CopyParamListView->ItemIndex,    CopyParamListView->ItemIndex + (Sender == UpCopyParamButton ? -1 : 1));}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::RemoveCopyParamButtonClick(  TObject * /*Sender*/){  DebugAssert(CopyParamListView->ItemIndex >= 1 &&    CopyParamListView->ItemIndex < (1 + FCopyParamList->Count));  FCopyParamList->Delete(CopyParamListView->ItemIndex - 1);  UpdateCopyParamListView();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddEditCopyParam(TCopyParamPresetMode Mode){  int Index = CopyParamListView->ItemIndex;  bool Result;  if ((Index == 0) && (Mode == cpmEdit))  {    Result = DoCopyParamCustomDialog(FCopyParams, 0);  }  else  {    TCopyParamRuleData * CopyParamRuleData =      (FDialogData != NULL ? FDialogData->CopyParamRuleData : NULL);    // negative (when default is selected) means add to the end    Index--;    TCopyParamType DefaultCopyParams;    // For cpmAdd use defaults.    if (Mode == cpmDuplicate)    {      // Only used, when duplicating default settings (Index < 0)      DefaultCopyParams = FCopyParams;    }    else if (Mode == cpmEdit)    {      // For cpmEdit, DefaultCopyParams is never used.      DebugAssert(Index >= 0);    }    Result = DoCopyParamPresetDialog(FCopyParamList, Index, Mode, CopyParamRuleData, DefaultCopyParams);    if (Result)    {      UpdateCopyParamListView();      CopyParamListView->ItemIndex = Index + 1;      // when using duplicate button, focus remains on original item      CopyParamListView->ItemFocused = CopyParamListView->Selected;    }  }  if (Result)  {    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddCopyParamButtonClick(TObject * /*Sender*/){  AddEditCopyParam(cpmAdd);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditCopyParamButtonClick(TObject * /*Sender*/){  AddEditCopyParam(cpmEdit);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::DuplicateCopyParamButtonClick(TObject * /*Sender*/){  AddEditCopyParam(cpmDuplicate);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamListViewDblClick(  TObject * /*Sender*/){  if (EditCopyParamButton->Enabled)  {    AddEditCopyParam(cpmEdit);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamListViewKeyDown(  TObject * /*Sender*/, WORD & Key, TShiftState /*Shift*/){  if (RemoveCopyParamButton->Enabled && (Key == VK_DELETE))  {    RemoveCopyParamButtonClick(NULL);  }  if (AddCopyParamButton->Enabled && (Key == VK_INSERT))  {    AddEditCopyParam(cpmAdd);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorMove(int Source, int Dest){  if (Source >= 0 && Source < FEditorList->Count &&      Dest >= 0 && Dest < FEditorList->Count)  {    FEditorList->Move(Source, Dest);    // workaround for bug in VCL    EditorListView3->ItemIndex = -1;    EditorListView3->ItemFocused = EditorListView3->Selected;    EditorListView3->ItemIndex = Dest;    UpdateEditorListView();    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorListView3DragDrop(TObject * Sender,  TObject * Source, int X, int Y){  if (Source == EditorListView3)  {    if (AllowListViewDrag(Sender, X, Y))    {      EditorMove(FListViewDragSource, FListViewDragDest);    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpDownEditorButtonClick(TObject *Sender){  EditorMove(EditorListView3->ItemIndex,    EditorListView3->ItemIndex + (Sender == UpEditorButton ? -1 : 1));}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::RemoveEditorButtonClick(  TObject * /*Sender*/){  DebugAssert(EditorListView3->ItemIndex >= 0 &&    EditorListView3->ItemIndex < FEditorList->Count);  FEditorList->Delete(EditorListView3->ItemIndex);  UpdateEditorListView();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddEditEditorButtonClick(TObject * Sender){  TEditorPreferencesMode Mode = (Sender == EditEditorButton ? epmEdit : epmAdd);  int Index = EditorListView3->ItemIndex;  TEditorPreferences * Editor;  if (Mode == epmEdit)  {    Editor = new TEditorPreferences(*FEditorList->Editors[Index]);  }  else  {    Editor = new TEditorPreferences();  }  try  {    bool DummyRemember = false;    if (DoEditorPreferencesDialog(Editor->GetData(), DummyRemember, Mode, true))    {      if (Mode == epmEdit)      {        FEditorList->Change(Index, Editor);      }      else      {        if (Index < 0)        {          Index = FEditorList->Count;          FEditorList->Add(Editor);        }        else        {          FEditorList->Insert(Index, Editor);        }      }      // ownership of the object lost      Editor = NULL;      UpdateEditorListView();      EditorListView3->ItemIndex = Index;      UpdateControls();    }  }  __finally  {    delete Editor;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorListView3DblClick(TObject * /*Sender*/){  if (EditEditorButton->Enabled)  {    AddEditEditorButtonClick(EditEditorButton);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorListView3KeyDown(TObject * /*Sender*/,  WORD & Key, TShiftState /*Shift*/){  if (RemoveEditorButton->Enabled && (Key == VK_DELETE))  {    RemoveEditorButtonClick(NULL);  }  if (AddEditorButton->Enabled && (Key == VK_INSERT))  {    AddEditEditorButtonClick(AddEditorButton);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdateEditorListView(){  EditorListView3->Items->Count = FEditorList->Count;  AutoSizeListColumnsWidth(EditorListView3);  EditorListView3->Invalidate();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorListView3Data(TObject * /*Sender*/,  TListItem * Item){  // WORKAROUND We get here on Wine after destructor is called  if (FEditorList != NULL)  {    int Index = Item->Index;    DebugAssert(Index >= 0 && Index <= FEditorList->Count);    const TEditorPreferences * Editor = FEditorList->Editors[Index];    Item->Caption = Editor->Name;    Item->SubItems->Add(Editor->Data->FileMask.Masks);    if (Editor->Data->Editor == edExternal)    {      Item->SubItems->Add(BooleanToStr(Editor->Data->ExternalEditorText));    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::NavigationTreeChange(TObject * /*Sender*/,  TTreeNode * Node){  if (DebugAlwaysTrue(Node->SelectedIndex > 0))  {    PageControl->ActivePage = DebugNotNull(FindPageForTreeNode(Node));    // reshow the accelerators, etc    ResetSystemSettings(this);    // This is particularly here to enable EditCopyParamButton,    // as for some reason CopyParamListView->Selected is NULL until    // its page is shown for the first time    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PageControlChange(TObject * /*Sender*/){  // this is probably only ever called from FormShow (explicitly)  bool Found = false;  if (DebugAlwaysTrue(PageControl->ActivePage->Tag > 0))  {    for (int Index = 0; Index < NavigationTree->Items->Count; Index++)    {      if (NavigationTree->Items->Item[Index]->SelectedIndex ==            PageControl->ActivePage->Tag)      {        NavigationTree->Items->Item[Index]->Selected = true;        Found = true;      }    }  }  if (DebugAlwaysTrue(Found))  {    UpdateControls();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CMDialogKey(TWMKeyDown & Message){  if (Message.CharCode == VK_TAB)  {    TShiftState Shift = KeyDataToShiftState(Message.KeyData);    if (Shift.Contains(ssCtrl))    {      TTreeNode * Node = NavigationTree->Selected;      if (!Shift.Contains(ssShift))      {        Node = Node->GetNext();        if (!Node) Node = NavigationTree->Items->GetFirstNode();      }      else      {        if (Node->GetPrev()) Node = Node->GetPrev();          else        while (Node->GetNext()) Node = Node->GetNext();      }      Node->Selected = True;      Message.Result = 1;      return;    }  }  TForm::Dispatch(&Message);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::WMHelp(TWMHelp & Message){  DebugAssert(Message.HelpInfo != NULL);  if (Message.HelpInfo->iContextType == HELPINFO_WINDOW)  {    // invoke help for active page (not for whole form), regardless of focus    // (e.g. even if focus is on control outside pagecontrol)    Message.HelpInfo->hItemHandle = PageControl->ActivePage->Handle;  }  TForm::Dispatch(&Message);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CMDpiChanged(TMessage & Message){  // To update font sizes - Note that they get scaled automatically, but as we use our own algorithm,  // we may end up using a slightly different size, so apply it straight away for consistency  UpdateControls();  TForm::Dispatch(&Message);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::Dispatch(void *Message){  TMessage * M = reinterpret_cast<TMessage*>(Message);  DebugAssert(M);  if (M->Msg == CM_DIALOGKEY)  {    CMDialogKey(*((TWMKeyDown *)Message));  }  else if (M->Msg == CM_DPICHANGED)  {    CMDpiChanged(*M);  }  else if (M->Msg == WM_HELP)  {    WMHelp(*((TWMHelp *)Message));  }  else  {    TForm::Dispatch(Message);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::RegisterAsUrlHandlersButtonClick(  TObject * /*Sender*/){  MenuPopup(RegisterAsUrlHandlerMenu, RegisterAsUrlHandlersButton);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::RegisterAsUrlHandlerItemClick(TObject * /*Sender*/){  unsigned int Result =    MessageDialog(MainInstructions(LoadStr(CONFIRM_REGISTER_URL2)),      qtConfirmation, qaYes | qaNo, HELP_REGISTER_URL);  if (Result == qaYes)  {    TInstantOperationVisualizer Visualizer;    RegisterForDefaultProtocols();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UnregisterForDefaultProtocolsItemClick(TObject * /*Sender*/){  unsigned int Result =    MessageDialog(MainInstructions(LoadStr(CONFIRM_UNREGISTER_URL)),      qtConfirmation, qaYes | qaNo, HELP_REGISTER_URL);  if (Result == qaYes)  {    TInstantOperationVisualizer Visualizer;    UnregisterForProtocols();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::MakeDefaultHandlerItemClick(TObject * /*Sender*/){  TOperationVisualizer Visualizer;  LaunchAdvancedAssociationUI();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::DDLabelClick(TObject * Sender){  ((Sender != DDFakeFileDisabledLabel) ? DDFakeFileEnabledButton : DDFakeFileDisabledButton)->SetFocus();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddSearchPathButtonClick(  TObject * /*Sender*/){  UnicodeString AppPath = ExtractFilePath(Application->ExeName);  if (MessageDialog(MainInstructions(FMTLOAD(CONFIRM_ADD_SEARCH_PATH, (AppPath))),        qtConfirmation, qaYes | qaNo, HELP_ADD_SEARCH_PATH) == qaYes)  {    TInstantOperationVisualizer Visualizer;    AddSearchPath(AppPath);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditorFontLabelDblClick(  TObject * Sender){  EditorFontButtonClick(Sender);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdateCopyParamListView(){  CopyParamListView->Items->Count = 1 + FCopyParamList->Count;  AutoSizeListColumnsWidth(CopyParamListView);  CopyParamListView->Invalidate();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamListViewData(TObject * /*Sender*/,  TListItem * Item){  // WORKAROUND We get here on Wine after destructor is called  if (FCopyParamList != NULL)  {    UnicodeString Name;    UnicodeString Rule;    int Index = Item->Index;    if (Index == 0)    {      Name = StripHotkey(LoadStr(COPY_PARAM_DEFAULT));    }    else    {      DebugAssert(Index >= 1 && Index <= 1 + FCopyParamList->Count);      Name = StripHotkey(FCopyParamList->Names[Index - 1]);      Rule = BooleanToStr(FCopyParamList->Rules[Index - 1] != NULL);    }    Item->Caption = Name;    Item->SubItems->Add(Rule);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::HelpButtonClick(TObject * /*Sender*/){  FormHelp(this);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PuttyPathBrowseButtonClick(  TObject * /*Sender*/){  UnicodeString Executables = FORMAT("%s;%s", (OriginalPuttyExecutable, KittyExecutable));  BrowseForExecutable(PuttyPathEdit, LoadStr(PREFERENCES_SELECT_PUTTY2),    FMTLOAD(PREFERENCES_PUTTY_FILTER2, (Executables)), false, false);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PathEditBeforeDialog(  TObject * /*Sender*/, UnicodeString & Name, bool & /*Action*/){  FBeforeDialogPath = Name;  Name = ExpandEnvironmentVariables(Name);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PathEditAfterDialog(  TObject * /*Sender*/, UnicodeString & Name, bool & /*Action*/){  if (IsPathToSameFile(Name, ExpandEnvironmentVariables(FBeforeDialogPath)))  {    Name = FBeforeDialogPath;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::NavigationTreeCollapsing(  TObject * /*Sender*/, TTreeNode * /*Node*/, bool & AllowCollapse){  AllowCollapse = false;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ListViewEndDrag(  TObject * Sender, TObject * /*Target*/, int /*X*/, int /*Y*/){  ScrollOnDragOver(Sender)->EndDrag();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::SessionReopenTimeoutEditSetValue(  TObject * /*Sender*/, Extended Value, UnicodeString & Text, bool & Handled){  if (Value == 0)  {    Text = LoadStr(PREFERENCES_RECONNECT_TIMEOUT_UNLIMITED);    Handled = true;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::SessionReopenTimeoutEditGetValue(  TObject * /*Sender*/, UnicodeString Text, Extended & Value, bool & Handled){  if (AnsiSameText(Text, LoadStr(PREFERENCES_RECONNECT_TIMEOUT_UNLIMITED)))  {    Value = 0;    Handled = true;  }}//---------------------------------------------------------------------------bool __fastcall TPreferencesDialog::CanSetMasterPassword(){  bool Result;  bool Retry;  do  {    Retry = false;    Result = !AnyOtherInstanceOfSelf();    if (!Result)    {      unsigned int Answer =        MessageDialog(          LoadStr(MASTER_PASSWORD_OTHER_INSTANCE),          qtConfirmation, qaRetry | qaIgnore | qaCancel,          HELP_MASTER_PASSWORD);      switch (Answer)      {        case qaRetry:          Retry = true;          break;        case qaIgnore:          Result = true;          break;        case qaCancel:        default:          // noop          break;      }    }  }  while (Retry && !Result);  return Result;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::MasterPasswordChanged(  UnicodeString Message, TStrings * RecryptPasswordErrors){  // Save master password.  // This is unlikely to fail, as storage is written explicitly already  // when writing the recrypted passwords  Configuration->SaveExplicit();  TQueryType QueryType = qtInformation;  if (RecryptPasswordErrors->Count > 0)  {    Message = FMTLOAD(MASTER_PASSWORD_RECRYPT_ERRORS, (Message));    QueryType = qtWarning;  }  MoreMessageDialog(    Message, RecryptPasswordErrors, QueryType, qaOK, HELP_MASTER_PASSWORD);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ChangeMasterPassword(UnicodeString Message){  UnicodeString NewPassword;  if (DoChangeMasterPasswordDialog(NewPassword))  {    std::unique_ptr<TStrings> RecryptPasswordErrors(new TStringList());    WinConfiguration->ChangeMasterPassword(NewPassword, RecryptPasswordErrors.get());    MasterPasswordChanged(Message, RecryptPasswordErrors.get());  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UseMasterPasswordCheckClick(  TObject * /*Sender*/){  if (UseMasterPasswordCheck->Checked != WinConfiguration->UseMasterPassword)  {    try    {      if (CanSetMasterPassword())      {        if (UseMasterPasswordCheck->Checked)        {          ChangeMasterPassword(LoadStr(MASTER_PASSWORD_SET2));        }        else        {          if (DoMasterPasswordDialog())          {            std::unique_ptr<TStrings> RecryptPasswordErrors(new TStringList());            WinConfiguration->ClearMasterPassword(RecryptPasswordErrors.get());            MasterPasswordChanged(LoadStr(MASTER_PASSWORD_CLEARED2), RecryptPasswordErrors.get());          }        }      }    }    __finally    {      UseMasterPasswordCheck->Checked = WinConfiguration->UseMasterPassword;      UpdateControls();    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::SetMasterPasswordButtonClick(  TObject * /*Sender*/){  if (CanSetMasterPassword())  {    ChangeMasterPassword(MainInstructions(LoadStr(MASTER_PASSWORD_CHANGED)));  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UsageViewButtonClick(TObject * /*Sender*/){  DoUsageStatisticsDialog();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamLabelClick(TObject * /*Sender*/){  if (EditCopyParamButton->Enabled)  {    AddEditCopyParam(cpmEdit);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CopyParamListViewCustomDrawItem(  TCustomListView * Sender, TListItem * Item,  TCustomDrawState /*State*/, bool & /*DefaultDraw*/){  if (Item->Index == 0)  {    Sender->Canvas->Font->Style = Sender->Canvas->Font->Style << fsBold;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::SelectPuttyRegistryStorageKey(const UnicodeString & Key){  PuttyRegistryStorageKeyEdit->ItemIndex =    PuttyRegistryStorageKeyEdit->Items->IndexOf(Key);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PuttyPathEditChange(TObject * /*Sender*/){  UnicodeString PuttyPath = PuttyPathEdit->Text;  if (ContainsText(PuttyPath, OriginalPuttyExecutable))  {    SelectPuttyRegistryStorageKey(OriginalPuttyRegistryStorageKey);  }  else if (ContainsText(PuttyPath, KittyExecutable))  {    SelectPuttyRegistryStorageKey(KittyRegistryStorageKey);  }  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::NavigationTreeChanging(TObject * /*Sender*/,  TTreeNode * Node, bool & /*AllowChange*/){  TTabSheet * Sheet = FindPageForTreeNode(Node);  // delay load as this can be time consuming  if (Sheet == LanguagesSheet)  {    LoadLanguages();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::LanguagesGetMoreButtonClick(TObject * /*Sender*/){  OpenBrowser(ProgramUrl(LoadStr(LOCALES_URL)));}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CommanderClick(TObject * /*Sender*/){  CommanderInterfaceButton2->SetFocus();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ExplorerClick(TObject * /*Sender*/){  ExplorerInterfaceButton2->SetFocus();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PanelFontLabelDblClick(TObject * Sender){  PanelFontButtonClick(Sender);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdatesAuthenticationEmailEditExit(TObject * /*Sender*/){  if (FVerifiedUpdatesAuthenticationEmail != UpdatesAuthenticationEmailEdit->Text)  {    if (!UpdatesAuthenticationEmailEdit->Text.IsEmpty())    {      TUpdatesConfiguration Updates = SaveUpdates();      {        TInstantOperationVisualizer Visualizer;        QueryUpdates(Updates);      }      UnicodeString AuthenticationError = Updates.Results.AuthenticationError;      if (!AuthenticationError.IsEmpty())      {        AuthenticationError = FormatUpdatesMessage(AuthenticationError);        if (HasParagraphs(AuthenticationError))        {          AuthenticationError = MainInstructionsFirstParagraph(AuthenticationError);        }        else        {          AuthenticationError = MainInstructions(AuthenticationError);        }        UnicodeString HelpUrl = GetEnableAutomaticUpdatesUrl();        unsigned int Result =          MoreMessageDialog(AuthenticationError, NULL, qtError, qaIgnore | qaAbort, HelpUrl);        if (Result == qaAbort)        {          Abort();        }      }    }    FVerifiedUpdatesAuthenticationEmail = UpdatesAuthenticationEmailEdit->Text;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::UpdatesLinkClick(TObject * /*Sender*/){  EnableAutomaticUpdates();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewWindowProc(TMessage & Message){  FOrigCustomCommandsViewWindowProc(Message);  if (Message.Msg == CN_NOTIFY)  {    TWMNotify & NotifyMessage = reinterpret_cast<TWMNotify &>(Message);    if (NotifyMessage.NMHdr->code == NM_CUSTOMDRAW)    {      // request CDDS_ITEMPOSTPAINT notification      Message.Result |= CDRF_NOTIFYPOSTPAINT | CDRF_NOTIFYSUBITEMDRAW;      TNMLVCustomDraw * CustomDraw = reinterpret_cast<TNMLVCustomDraw *>(NotifyMessage.NMHdr);      int Index = CustomDraw->nmcd.dwItemSpec;      int CommandIndex = GetCommandIndex(Index);      TCustomCommandList * List = GetCommandList(Index);      // after end of every list, except for the last last list      if ((CommandIndex == List->Count - 1) && (Index < CustomCommandsView->Items->Count - 1) &&          FLAGSET(CustomDraw->nmcd.dwDrawStage, CDDS_ITEMPOSTPAINT))      {        TRect Rect;        Rect.Top = CustomDraw->iSubItem;        Rect.Left = LVIR_BOUNDS;        CustomCommandsView->Perform(LVM_GETSUBITEMRECT, CustomDraw->nmcd.dwItemSpec, reinterpret_cast<LPARAM>(&Rect));        HDC DC = CustomDraw->nmcd.hdc;        SelectObject(DC, GetStockObject(DC_PEN));        SetDCPenColor(DC, ColorToRGB(clWindowFrame));        MoveToEx(DC, Rect.Left, Rect.Bottom - 1, NULL);        LineTo(DC, Rect.Right, Rect.Bottom - 1);      }    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ExtensionHttpError(THttp * /*Sender*/, int Status, const UnicodeString & Message){  if ((Status / 10) == 49)  {    // HTTP 49x indicate user-friendly error message from winscp.net, throw it without HTTP status code    throw Exception(Message);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddExtension(){  const UnicodeString HistoryKey(L"ExtensionPath");  std::unique_ptr<TStrings> History(CloneStrings(CustomWinConfiguration->History[HistoryKey]));  UnicodeString Path;  if (InputDialog(LoadStr(ADD_EXTENSION_CAPTION), LoadStr(ADD_EXTENSION_PROMPT), Path,        HELP_NONE, History.get(), true) &&      !Path.IsEmpty())  {    CustomWinConfiguration->History[HistoryKey] = History.get();    bool Trusted;    bool Latest;    UnicodeString FileName;    UnicodeString ExtensionPath;    std::unique_ptr<TStringList> Lines(new TStringList());    std::unique_ptr<TCustomCommandType> CustomCommand;    bool IsUrl = IsHttpOrHttpsUrl(Path);    try    {      UnicodeString ProvisionaryId;      if (IsUrl)      {        UnicodeString Url = Path;        Url = SecureUrl(Url);        bool WinSCPURL = IsWinSCPUrl(Url);        if (WinSCPURL)        {          Url = ProgramUrl(Url);          // The EncodeUrlString should not be necessary, but as we get the value from registry, let's be safe          Url = AppendUrlParams(Url, FORMAT(L"netframework=%s", (EncodeUrlString(GetNetVersionStr()))));          Url = AppendUrlParams(Url, FORMAT(L"powershell=%s", (EncodeUrlString(GetPowerShellVersionStr()))));          Url = AppendUrlParams(Url, FORMAT(L"windows=%s", (EncodeUrlString(WindowsVersion()))));          Url = CampaignUrl(Url);        }        TOperationVisualizer Visualizer;        std::unique_ptr<THttp> Http(CreateHttp());        Http->URL = Url;        std::unique_ptr<TStrings> Headers(new TStringList());        Headers->Values[L"Accept"] = L"text/winscpextension,text/plain";        Http->RequestHeaders = Headers.get();        Http->OnError = ExtensionHttpError;        Http->Get();        UnicodeString TrustedStr = Http->ResponseHeaders->Values[L"WinSCP-Extension-Trusted"];        Trusted = WinSCPURL && (StrToIntDef(TrustedStr, 0) != 0);        FileName = MakeValidFileName(Http->ResponseHeaders->Values[L"WinSCP-Extension-Id"]);        if (FileName.IsEmpty())        {          FileName = MakeValidFileName(ExtractFileNameFromUrl(Path));        }        ProvisionaryId = WinConfiguration->GetProvisionaryExtensionId(FileName);        Lines->Text = Http->Response;        Latest = Http->ResponseHeaders->Values[L"WinSCP-Extension-Skipped"].Trim().IsEmpty();      }      else      {        if (!FileExists(ApiPath(Path)))        {          throw Exception(MainInstructions(FMTLOAD(FILE_NOT_EXISTS, (Path))));        }        Trusted = true;        Latest = true;        UnicodeString Id = WinConfiguration->GetExtensionId(Path);        FileName = ExtractFileName(Path);        if (!Id.IsEmpty())        {          ExtensionPath = Path;          ProvisionaryId = Id;        }        else        {          ProvisionaryId = WinConfiguration->GetProvisionaryExtensionId(FileName);        }        LoadScriptFromFile(Path, Lines.get());      }      // validate syntax      CustomCommand.reset(new TCustomCommandType());      // Provisionary Id, just for the ExtensionStringTranslation, so that the test for EXTENSION_DUPLICATE below works      CustomCommand->Id = ProvisionaryId;      CustomCommand->LoadExtension(Lines.get(), FileName);    }    catch (Exception & E)    {      throw ExtException(&E, MainInstructions(FMTLOAD(EXTENSION_LOAD_ERROR, (Path))));    }    if (!ExtensionPath.IsEmpty())    {      int Index = FExtensionList->FindIndexByFileName(Path);      if (Index >= 0)      {        CustomCommandsView->ItemIndex = GetCommandListIndex(FExtensionList, Index);        CustomCommandsView->ItemFocused->MakeVisible(false);        CustomCommandsView->SetFocus();        throw Exception(MainInstructions(LoadStr(EXTENSION_INSTALLED_ALREADY)));      }    }    if (FExtensionList->Find(CustomCommand->Name) != NULL)    {      throw Exception(MainInstructions(FMTLOAD(EXTENSION_DUPLICATE, (StripHotkey(CustomCommand->Name)))));    }    if (ExtensionPath.IsEmpty())    {      if (TCustomCommandType::GetExtensionId(FileName).IsEmpty())      {        UnicodeString FileNameOnly = ExtractFileNameOnly(FileName);        if (FileNameOnly.IsEmpty())        {          FileName = MakeValidFileName(StripHotkey(CustomCommand->Name)) + WinSCPExtensionExt;        }        else        {          FileName = ExtractFileNameOnly(FileName) + WinSCPExtensionExt + ExtractFileExt(FileName);        }      }    }    if (Trusted ||        (MessageDialog(MainInstructions(LoadStr(EXTENSION_UNTRUSTED)), qtWarning, qaOK | qaCancel) == qaOK))    {      if (ExtensionPath.IsEmpty())      {        UnicodeString UserExtensionsPath = WinConfiguration->GetUserExtensionsPath();        if (!DirectoryExists(UserExtensionsPath) &&            !ForceDirectories(UserExtensionsPath))        {          throw EOSExtException(MainInstructions(FMTLOAD(CREATE_LOCAL_DIR_ERROR, (UserExtensionsPath))));        }        ExtensionPath = IncludeTrailingBackslash(UserExtensionsPath) + FileName;        int Counter = 1;        UnicodeString OriginalExtensionPath = ExtensionPath;        int P = Pos(UpperCase(WinSCPExtensionExt), UpperCase(OriginalExtensionPath));        while (FileExists(ApiPath(ExtensionPath)))        {          Counter++;          ExtensionPath = WinConfiguration->UniqueExtensionName(LeftStr(OriginalExtensionPath, P - 1), Counter) + RightStr(OriginalExtensionPath, OriginalExtensionPath.Length() - P + 1);        }        Lines->SaveToFile(ApiPath(ExtensionPath));        FAddedExtensions->Add(ExtensionPath);      }      int Index = GetListCommandIndex(FExtensionList);      std::unique_ptr<TCustomCommandType> CustomCommand(new TCustomCommandType());      CustomCommand->Id = WinConfiguration->GetExtensionId(ExtensionPath);      CustomCommand->LoadExtension(ExtensionPath);      Index = AddCommandToList(FExtensionList, Index, CustomCommand.release());      UpdateCustomCommandsView();      CustomCommandsView->ItemIndex = GetCommandListIndex(FExtensionList, Index);      CustomCommandsView->ItemFocused->MakeVisible(false);      UpdateControls();      if (!Latest)      {        MessageDialog(LoadStr(EXTENSION_NOT_LATEST), qtInformation, qaOK);      }      if (IsUrl)      {        Configuration->Usage->Inc(L"ExtensionAddsFromUrl");      }      else      {        Configuration->Usage->Inc(L"ExtensionAddsFromFile");      }    }  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddCommandButtonClick(TObject * /*Sender*/){  if (GetCommandList(CustomCommandsView->ItemIndex) == FCustomCommandList)  {    AddEditCommand(false);  }  else  {    AddExtension();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddCustomCommandMenuItemClick(TObject * /*Sender*/){  AddEditCommand(false);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddExtensionMenuItemClick(TObject * /*Sender*/){  AddExtension();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::EditCommandButtonClick(TObject * /*Sender*/){  AddEditCommand(true);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AddCommandButtonDropDownClick(TObject * /*Sender*/){  AddCustomCommandMenuItem->Default = (GetCommandList(CustomCommandsView->ItemIndex) == FCustomCommandList);  AddExtensionMenuItem->Default = (GetCommandList(CustomCommandsView->ItemIndex) == FExtensionList);  MenuPopup(AddCommandMenu, AddCommandButton);}//---------------------------------------------------------------------------UnicodeString __fastcall TPreferencesDialog::GetSessionKey(){  TTerminal * Terminal = TTerminalManager::Instance()->ActiveTerminal;  UnicodeString Result;  if (Terminal != NULL)  {    Result = Terminal->SessionData->SessionKey;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomCommandsViewMouseMove(TObject * /*Sender*/, TShiftState /*Shift*/, int X, int Y){  TListItem * Item = CustomCommandsView->GetItemAt(X, Y);  int Index = (Item != NULL) ? Item->Index : -1;  if (Index != FCustomCommandsHintItem)  {    Application->CancelHint();    UnicodeString Hint;    if (Index >= 0)    {      TCustomCommandList * List = GetCommandList(Index);      const TCustomCommandType * Command = List->Commands[GetCommandIndex(Index)];      Hint = StripHotkey(Command->Name);      if (Command->ShortCut != 0)      {        Hint = FORMAT(L"%s (%s)", (Hint, ShortCutToText(Command->ShortCut)));      }      if (!Command->Description.IsEmpty())      {        Hint += L"\n" + Command->Description;      }      Hint += L"\n" + Command->GetCommandWithExpandedOptions(FCustomCommandOptions.get(), GetSessionKey());      if (List == FExtensionList)      {        Hint += L"\n" + Command->FileName;      }    }    CustomCommandsView->Hint = Hint;    FCustomCommandsHintItem = Index;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::BackgroundConfirmationsLinkClick(TObject * /*Sender*/){  PageControl->ActivePage = QueueSheet;  PageControlChange(NULL);  QueueNoConfirmationCheck->SetFocus();  QueueNoConfirmationCheck->Perform(WM_CHANGEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ConfigureCommandButtonClick(TObject * /*Sender*/){  ConfigureCommand();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::ConfigureCommand(){  int Index = CustomCommandsView->ItemIndex;  TCustomCommandList * CommandList = GetCommandList(Index);  int CommandIndex = GetCommandIndex(Index);  const TCustomCommandType * Command = CommandList->Commands[CommandIndex];  UnicodeString Site = GetSessionKey();  if (Command->AnyOptionWithFlag(TCustomCommandType::ofSite) &&      Site.IsEmpty())  {    throw Exception(LoadStr(NO_SITE_FOR_COMMAND));  }  TShortCut ShortCut = Command->ShortCut;  TShortCuts ShortCuts = GetShortCuts();  if (DoCustomCommandOptionsDialog(        Command, FCustomCommandOptions.get(), &ShortCut, TCustomCommandType::ofConfig, NULL, GetSessionKey(), &ShortCuts))  {    TCustomCommandType * UpdatedCommand = new TCustomCommandType(*Command);    UpdatedCommand->ShortCut = ShortCut;    CommandList->Change(CommandIndex, UpdatedCommand);    UpdateCustomCommandsView();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::LanguagesViewCustomDrawItem(  TCustomListView * Sender, TListItem * Item, TCustomDrawState /*State*/, bool & /*DefaultDraw*/){  TLocaleInfo * LocaleInfo = static_cast<TLocaleInfo *>(Item->Data);  if (LocaleInfo->Locale == WinConfiguration->DefaultLocale)  {    Sender->Canvas->Font->Style = Sender->Canvas->Font->Style << fsBold;  }  if (LocaleInfo->Completeness < 0)  {    Sender->Canvas->Font->Color = clRed;  }  else if (LocaleInfo->Completeness < 100)  {    Sender->Canvas->Font->Color = clGrayText;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::LogMaxSizeComboExit(TObject * /*Sender*/){  __int64 Size;  if (!TryStrToSize(LogMaxSizeCombo->Text, Size))  {    LogMaxSizeCombo->SetFocus();    throw Exception(FMTLOAD(SIZE_INVALID, (LogMaxSizeCombo->Text)));  }  else  {    LogMaxSizeCombo->Text = SizeToStr(Size);  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::PuttyPathEditExit(TObject * /*Sender*/){  try  {    UnicodeString Program, AParams, Dir;    SplitCommand(PuttyPathEdit->Text, Program, AParams, Dir);  }  catch(...)  {    PuttyPathEdit->SelectAll();    PuttyPathEdit->SetFocus();    throw;  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::AutomaticIniFileStorageLabelGetStatus(TCustomPathLabel * /*Sender*/, bool & Active){  Active = AutomaticIniFileStorageButton->Checked;}//---------------------------------------------------------------------------UnicodeString __fastcall TPreferencesDialog::GetCustomIniFileStorageName(){  UnicodeString Result;  if (CustomIniFileStorageButton->Checked)  {    Result = CustomIniFileStorageEdit->Text;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomIniFileStorageChanged(){  UnicodeString CustomIniFileStorageName = GetCustomIniFileStorageName();  if (!CustomIniFileStorageName.IsEmpty() &&      !IsPathToSameFile(CustomIniFileStorageName, FCustomIniFileStorageName) &&      FileExists(CustomIniFileStorageName))  {    UnicodeString Message = FORMAT(LoadStrPart(CUSTOM_INI_FILE_OVERWRITE, 1), (CustomIniFileStorageName));    TMessageParams Params;    TQueryButtonAlias Aliases[2];    Aliases[0].Button = qaYes;    Aliases[0].Alias = LoadStrPart(CUSTOM_INI_FILE_OVERWRITE, 2);    Aliases[1].Button = qaNo;    Aliases[1].Alias = LoadStrPart(CUSTOM_INI_FILE_OVERWRITE, 3);    Params.Aliases = Aliases;    Params.AliasesCount = 2;    unsigned int Result = MessageDialog(Message, qtConfirmation, qaYes | qaNo | qaCancel, HELP_MOVE_CONFIGURATION, &Params);    if (Result == qaYes)    {      // noop    }    else if (Result == qaNo)    {      Configuration->ScheduleCustomIniFileStorageUse(GetCustomIniFileStorageName());      ExecuteNewInstance(L"");      TerminateApplication();    }    else    {      Abort();    }  }  FCustomIniFileStorageName = CustomIniFileStorageName;}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomIniFileStorageEditExit(TObject * /*Sender*/){  CustomIniFileStorageChanged();}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomIniFileStorageEditAfterDialog(TObject * Sender, UnicodeString & Name, bool & Action){  PathEditAfterDialog(Sender, Name, Action);  if (Action)  {    CustomIniFileStorageEdit->Text = Name;    CustomIniFileStorageChanged();  }}//---------------------------------------------------------------------------void __fastcall TPreferencesDialog::CustomIniFileStorageButtonClick(TObject * /*Sender*/){  UpdateControls();  // Handler is shown also when Checked is set from LoadConfiguration  if (FNoUpdate == 0)  {    // Focus to force validation    CustomIniFileStorageEdit->SetFocus();  }}//---------------------------------------------------------------------------
 |