| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663 | //---------------------------------------------------------------------#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"//---------------------------------------------------------------------#pragma link "CopyParams"#pragma link "UpDownEdit"#pragma link "ComboEdit"#ifndef NO_RESOURCES#pragma link "HistoryComboBox"#pragma resource "*.dfm"#endif//---------------------------------------------------------------------bool __fastcall DoPreferencesDialog(TPreferencesMode APreferencesMode,  TPreferencesDialogData * DialogData){  bool Result;  TPreferencesDialog * PreferencesDialog =    new TPreferencesDialog(GetFormOwner(), APreferencesMode);  try  {    Result = PreferencesDialog->Execute(DialogData);    if (Result)    {      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);  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);  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();    TStrings * Locales = GUIConfiguration->Locales;    for (int Index = 0; Index < Locales->Count; Index++)    {      TListItem * Item = LanguagesView->Items->Add();      Item->Caption = Locales->Strings[Index];      Item->Data = Locales->Objects[Index];      Item->Focused =        (reinterpret_cast<LCID>(Locales->Objects[Index]) == GUIConfiguration->Locale);      Item->Selected = Item->Focused;    }    FLanguagesLoaded = false;  }}//---------------------------------------------------------------------------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(DDTransferConfirmation);    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(DDAllowMoveInit);    BOOLPROP(BeepOnFinish);    BOOLPROP(TemporaryDirectoryAppendSession);    BOOLPROP(TemporaryDirectoryAppendPath);    BOOLPROP(TemporaryDirectoryDeterministic);    BOOLPROP(TemporaryDirectoryCleanup);    BOOLPROP(ConfirmTemporaryDirectoryCleanup);    BOOLPROP(FullRowSelect);    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      DDTransferConfirmationCheck->AllowGrayed = true;    }    CheckBoxAutoSwitchLoad(DDTransferConfirmationCheck, WinConfiguration->DDTransferConfirmation);    BeepOnFinishAfterEdit->AsInteger =      int(static_cast<double>(GUIConfiguration->BeepOnFinishAfter) * SecsPerDay);    BOOLPROP(BalloonNotifications);    DDExtEnabledButton->Checked = WinConfiguration->DDExtEnabled;    DDExtDisabledButton->Checked = !DDExtEnabledButton->Checked;    DDWarnOnMoveCheck->Checked = !WinConfiguration->DDAllowMove;    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);    IniFileStorageButton2->Checked = (Configuration->Storage == stIniFile);    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;    QueueIndividuallyCheck->Checked = GUIConfiguration->DefaultCopyParam.QueueIndividually;    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    {      FPanelFont->Assign(WinConfiguration->SystemIconFont);    }    // 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;    LogToFileCheck->Checked = Configuration->LogToFile;    LogFileNameEdit3->Text =      !Configuration->LogFileName.IsEmpty() ? Configuration->LogFileName : Configuration->DefaultLogFileName;    if (Configuration->LogFileAppend)    {      LogFileAppendButton->Checked = true;    }    else    {      LogFileOverwriteButton->Checked = true;    }    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(DDAllowMoveInit);    BOOLPROP(BeepOnFinish);    BOOLPROP(TemporaryDirectoryAppendSession);    BOOLPROP(TemporaryDirectoryAppendPath);    BOOLPROP(TemporaryDirectoryDeterministic);    BOOLPROP(TemporaryDirectoryCleanup);    BOOLPROP(ConfirmTemporaryDirectoryCleanup);    BOOLPROP(FullRowSelect);    WinConfiguration->ConfirmClosingSession = ConfirmClosingSessionCheck2->Checked;    WinConfiguration->DDTransferConfirmation =      CheckBoxAutoSwitchSave(DDTransferConfirmationCheck);    GUIConfiguration->BeepOnFinishAfter =      static_cast<double>(BeepOnFinishAfterEdit->Value / SecsPerDay);    BOOLPROP(BalloonNotifications);    WinConfiguration->DDAllowMove = !DDWarnOnMoveCheck->Checked;    WinConfiguration->DDExtEnabled = DDExtEnabledButton->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.QueueIndividually = QueueIndividuallyCheck->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;    Configuration->LogProtocol = LogProtocolCombo->ItemIndex;    Configuration->LogFileName = LogToFileCheck->Checked ? LogFileNameEdit3->Text : UnicodeString();    Configuration->LogFileAppend = LogFileAppendButton->Checked;    BOOLPROP(LogSensitive);    Configuration->LogActions = EnableActionsLoggingCheck->Checked;    Configuration->ActionsLogFileName = ActionsLogFileNameEdit->Text;    // languages    // As this dialog does not explicitly support run-time locale changing,    // make this last, otherwise we lose some settings (or even worse    // we end-up saving default text box values=control names as our configuration)    if (LanguagesView->ItemFocused != NULL)    {      GUIConfiguration->Locale =        reinterpret_cast<LCID>(LanguagesView->ItemFocused->Data);    }    // This possibly fails, make it last, so that the other settings are preserved.    // Do nothing when no option is selected (i.e. storage is stNul).    if (RegistryStorageButton->Checked)    {      Configuration->Storage = stRegistry;    }    else if (IniFileStorageButton2->Checked)    {      Configuration->Storage = stIniFile;    }    #undef BOOLPROP  }  __finally  {    Configuration->EndUpdate();  }}//---------------------------------------------------------------------------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(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() : WinConfiguration->SystemIconFont;    UnicodeString PanelFontLabelText;    PanelFontLabelText = FMTLOAD(EDITOR_FONT_FMT,      (ActualPanelFont->Name, ActualPanelFont->Size));    PanelFontLabel->Caption = PanelFontLabelText;    if (!SameFont(PanelFontLabel->Font, ActualPanelFont))    {      PanelFontLabel->Font = ActualPanelFont;    }    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;    TColor EditorFontColor = GetWindowTextColor(FEditorFont->Color);    if (!SameFont(EditorFontLabel->Font, FEditorFont.get()) ||        (EditorFontLabel->Font->Color != EditorFontColor))    {      std::unique_ptr<TFont> Font(new TFont);      Font->Assign(FEditorFont.get());      Font->Color = EditorFontColor;      EditorFontLabel->Font = Font.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 && CommandList->Commands[CommandIndex]->AnyOptionWithFlag(TCustomCommandType::ofConfig));    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);    EnableControl(DDExtEnabledButton, WinConfiguration->DDExtInstalled);    EnableControl(DDExtEnabledLabel, WinConfiguration->DDExtInstalled);    EnableControl(DDExtDisabledPanel, DDExtDisabledButton->Checked);    EnableControl(DDTemporaryDirectoryEdit, DDCustomTemporaryDirectoryButton->Enabled &&      DDCustomTemporaryDirectoryButton->Checked);    EnableControl(DDWarnOnMoveCheck, DDExtDisabledButton->Checked &&      DDAllowMoveInitCheck->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 || IniFileStorageButton2->Checked);    IniFileStorageButton2->Caption =      AnsiReplaceStr(IniFileStorageButton2->Caption, L"(winscp.ini)",        FORMAT(L"(%s)", (ExpandEnvironmentVariables(Configuration->IniFileStorageName))));    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(UsageViewButton, CollectUsageCheck->Checked);    EnableControl(UpdatesProxyHostEdit, UpdatesProxyCheck->Checked);    EnableControl(UpdatesProxyHostLabel, UpdatesProxyHostEdit->Enabled);    EnableControl(UpdatesProxyPortEdit, UpdatesProxyCheck->Checked);    EnableControl(UpdatesProxyPortLabel, UpdatesProxyPortEdit->Enabled);    bool IsSiteCommand = false;    try    {      TRemoteCustomCommand RemoteCustomCommand;      TInteractiveCustomCommand InteractiveCustomCommand(&RemoteCustomCommand);      UnicodeString PuttyPath = PuttyPathEdit->Text;      PuttyPath = InteractiveCustomCommand.Complete(PuttyPath, false);      IsSiteCommand = RemoteCustomCommand.IsSiteCommand(PuttyPath);    }    catch (...)    {      // noop    }    EnableControl(PuttyPasswordCheck2, !PuttyPathEdit->Text.IsEmpty());    EnableControl(AutoOpenInPuttyCheck, PuttyPasswordCheck2->Enabled);    EnableControl(TelnetForFtpInPuttyCheck,      PuttyPasswordCheck2->Enabled && !IsSiteCommand);    EnableControl(PuttyRegistryStorageKeyEdit,      PuttyPasswordCheck2->Enabled && !IsSiteCommand);    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    // There's no quick launch in Windows 7    EnableControl(QuickLaunchIconButton, !IsWin7());    MakeDefaultHandlerItem->Visible = IsWinVista();    // languages    LanguageChangeLabel->Visible =      DebugAlwaysTrue(!GUIConfiguration->CanApplyLocaleImmediately) &&      (LanguagesView->ItemFocused != NULL) &&      (reinterpret_cast<LCID>(LanguagesView->ItemFocused->Data) != GUIConfiguration->AppliedLocale);    // logging    EnableControl(LogProtocolCombo, EnableLoggingCheck->Checked);    EnableControl(LogToFileCheck, LogProtocolCombo->Enabled);    EnableControl(LogFileNameEdit3, LogToFileCheck->Enabled && LogToFileCheck->Checked);    EnableControl(LogFileNameHintText, LogFileNameEdit3->Enabled);    EnableControl(LogFileAppendButton, LogFileNameEdit3->Enabled);    EnableControl(LogFileOverwriteButton, LogFileNameEdit3->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  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;}//---------------------------------------------------------------------------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;  if (WinConfiguration->SharedBookmarks != NULL)  {    WinConfiguration->SharedBookmarks->ShortCuts(ShortCuts);  }  FCustomCommandList->ShortCuts(ShortCuts);  FExtensionList->ShortCuts(ShortCuts);  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--;    Result = DoCopyParamPresetDialog(FCopyParamList, Index, Mode, CopyParamRuleData, FCopyParams);    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::Dispatch(void *Message){  TMessage * M = reinterpret_cast<TMessage*>(Message);  DebugAssert(M);  if (M->Msg == CM_DIALOGKEY)  {    CMDialogKey(*((TWMKeyDown *)Message));  }  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::DDExtLabelClick(TObject * Sender){  ((Sender == DDExtEnabledLabel) ? DDExtEnabledButton : DDExtDisabledButton)->    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 (CompareFileName(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*/){  std::unique_ptr<TStrings> Data(TextToStringList(GetUsageData()));  UnicodeString Message =    Data->Text.IsEmpty() ? MainInstructions(LoadStr(USAGE_DATA_NONE)) : LoadStr(USAGE_DATA2);  MoreMessageDialog(Message, Data.get(), qtInformation, qaOK, HELP_USAGE);}//---------------------------------------------------------------------------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)  {    FVerifiedUpdatesAuthenticationEmail = UpdatesAuthenticationEmailEdit->Text;    if (!FVerifiedUpdatesAuthenticationEmail.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();        }      }    }  }}//---------------------------------------------------------------------------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::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    {      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 regitry, 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->Get();        UnicodeString TrustedStr = Http->ResponseHeaders->Values["WinSCP-Extension-Trusted"];        Trusted = WinSCPURL && (StrToIntDef(TrustedStr, 0) != 0);        FileName = MakeValidFileName(Http->ResponseHeaders->Values["WinSCP-Extension-Id"]);        if (FileName.IsEmpty())        {          FileName = MakeValidFileName(ExtractFileNameFromUrl(Path));        }        Lines->Text = Http->Response;        Latest = Http->ResponseHeaders->Values["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;        }        LoadScriptFromFile(Path, Lines.get());      }      // validate syntax      CustomCommand.reset(new TCustomCommandType());      CustomCommand->LoadExtension(Lines.get());    }    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 = LeftStr(OriginalExtensionPath, P - 1) + IntToStr(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);}//---------------------------------------------------------------------------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());      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;  const TCustomCommandType * Command = GetCommandList(Index)->Commands[GetCommandIndex(Index)];  DoCustomCommandOptionsDialog(Command, FCustomCommandOptions.get(), TCustomCommandType::ofConfig);  UpdateCustomCommandsView();}//---------------------------------------------------------------------------
 |