| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331 | //---------------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include "ScpCommander.h"#include <Common.h>#include <CoreMain.h>#include <Interface.h>#include <TextsCore.h>#include <TextsWin.h>#include <HelpWin.h>#include <VCLCommon.h>#include <GUITools.h>#include <DragDrop.hpp>#include <StrUtils.hpp>#include "Glyphs.h"#include "NonVisual.h"#include "Tools.h"#include "WinConfiguration.h"#include "TerminalManager.h"#include "Bookmarks.h"//---------------------------------------------------------------------------#pragma package(smart_init)#pragma link "CustomDirView"#pragma link "CustomScpExplorer"#pragma link "CustomUnixDirView"#pragma link "IEListView"#pragma link "NortonLikeListView"#pragma link "UnixDirView"#pragma link "DirView"#pragma link "PathLabel"#pragma link "HistoryComboBox"#pragma link "CustomDriveView"#pragma link "DriveView"#pragma link "UnixDriveView"#pragma link "TB2Dock"#pragma link "TB2Item"#pragma link "TB2Toolbar"#pragma link "TBX"#pragma link "TB2ExtItems"#pragma link "TBXExtItems"#pragma link "TBXLists"#pragma link "TBXStatusBars"#pragma link "TBXToolPals"#pragma link "ThemePageControl"#pragma resource "*.dfm"//---------------------------------------------------------------------------class TSynchronizedBrowsingGuard{public:  TSynchronizedBrowsingGuard()  {    FWasSynchronisingBrowsing = NonVisualDataModule->SynchronizeBrowsingAction->Checked;    NonVisualDataModule->SynchronizeBrowsingAction->Checked = false;  }  ~TSynchronizedBrowsingGuard()  {    NonVisualDataModule->SynchronizeBrowsingAction->Checked = FWasSynchronisingBrowsing;  }private:  bool FWasSynchronisingBrowsing;};//---------------------------------------------------------------------------__fastcall TScpCommanderForm::TScpCommanderForm(TComponent* Owner)        : TCustomScpExplorerForm(Owner){  FConstructed = true;  FCurrentSide = osLocal;  FLastLeftPanelWidth = LeftPanelWidth;  FNormalPanelsWidth = -1;  FSynchronisingBrowse = false;  FFirstTerminal = true;  FInternalDDDownloadList = new TStringList();  FLocalPathComboBoxPaths = new TStringList();  LocalPathComboUpdateDrives();  LocalBackButton->LinkSubitems = HistoryMenu(osLocal, true)->Items;  LocalForwardButton->LinkSubitems = HistoryMenu(osLocal, false)->Items;  RemoteBackButton->LinkSubitems = HistoryMenu(osRemote, true)->Items;  RemoteForwardButton->LinkSubitems = HistoryMenu(osRemote, false)->Items;  TopDock->PopupMenu = NonVisualDataModule->CommanderBarPopup;  CopyPopup(StatusBar, TopDock);  CopyPopup(QueueDock, TopDock);  CopyPopup(QueueLabel, TopDock);  CopyPopup(BottomDock, TopDock);  CopyPopup(QueueSeparatorPanel, TopDock);  CopyPopup(QueueFileList, TopDock);  CopyPopup(QueueFileListSplitter, TopDock);  LocalTopDock->PopupMenu = NonVisualDataModule->LocalPanelPopup;  CopyPopup(LocalPathLabel, LocalTopDock);  CopyPopup(LocalStatusBar, LocalTopDock);  CopyPopup(LocalDriveView, LocalTopDock);  CopyPopup(LocalBottomDock, LocalTopDock);  RemoteTopDock->PopupMenu = NonVisualDataModule->RemotePanelPopup;  CopyPopup(RemotePathLabel, RemoteTopDock);  CopyPopup(RemoteStatusBar, RemoteTopDock);  CopyPopup(RemoteDriveView, RemoteTopDock);  CopyPopup(RemoteBottomDock, RemoteTopDock);  SetShortcuts();  Splitter->ShowHint = True;  LocalPanelSplitter->ShowHint = true;  reinterpret_cast<TLabel*>(Splitter)->OnDblClick = SplitterDblClick;  reinterpret_cast<TLabel*>(LocalPanelSplitter)->OnDblClick = PanelSplitterDblClick;  reinterpret_cast<TLabel*>(RemotePanelSplitter)->OnDblClick = PanelSplitterDblClick;  CommandLineCombo->Text = L"";  FCommandLineComboPopulated = false;  for (int i = 0; i < Toolbar2Toolbar->Items->Count; i++)  {    UpdateToolbar2ItemCaption(Toolbar2Toolbar->Items->Items[i]);  }  UseDesktopFont(LocalDirView);  UseDesktopFont(LocalDriveView);  UseDesktopFont(LocalPathLabel);  UseDesktopFont(RemotePathLabel);  UseDesktopFont(LocalStatusBar);  UseDesktopFont(StatusBar);  NonVisualDataModule->QueueSpeedComboBoxItem(QueueSpeedComboBoxItem);  // particularly to reorder panels on right-to-left bidi mode  ConfigurationChanged();}//---------------------------------------------------------------------------__fastcall TScpCommanderForm::~TScpCommanderForm(){  delete FInternalDDDownloadList;  SAFE_DESTROY(FLocalPathComboBoxPaths);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdateToolbar2ItemCaption(TTBCustomItem * Item){  Item->Caption =    ShortCutToText(Item->ShortCut) + L" " +    StripEllipsis(StripHotkey(Item->Caption));}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RestoreFormParams(){  DebugAssert(WinConfiguration);  TCustomScpExplorerForm::RestoreFormParams();  RestoreForm(WinConfiguration->ScpCommander.WindowParams, this);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RestorePanelParams(  TCustomDirView * DirView, TControl * DriveControl, TTBXStatusBar * StatusBar,  const TScpCommanderPanelConfiguration & PanelConfiguration){  DirView->ColProperties->ParamsStr = PanelConfiguration.DirViewParams;  StatusBar->Visible = PanelConfiguration.StatusBar;  DriveControl->Visible = PanelConfiguration.DriveView;  if (DriveControl->Align == alTop)  {    DriveControl->Height = LoadDimension(PanelConfiguration.DriveViewHeight, PanelConfiguration.DriveViewHeightPixelsPerInch, this);  }  else  {    DriveControl->Width = LoadDimension(PanelConfiguration.DriveViewWidth, PanelConfiguration.DriveViewWidthPixelsPerInch, this);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RestoreParams(){  DebugAssert(Configuration);  TCustomScpExplorerForm::RestoreParams();  LeftPanelWidth = WinConfiguration->ScpCommander.LocalPanelWidth;  LoadToolbarsLayoutStr(WinConfiguration->ScpCommander.ToolbarsLayout, WinConfiguration->ScpCommander.ToolbarsButtons);  if (IsUWP())  {    UpdatesToolbar->Visible = false;  }  SessionsPageControl->Visible = WinConfiguration->ScpCommander.SessionsTabs;  StatusBar->Visible = WinConfiguration->ScpCommander.StatusBar;  RestorePanelParams(LocalDirView, LocalDriveView, LocalStatusBar, WinConfiguration->ScpCommander.LocalPanel);  RestorePanelParams(RemoteDirView, RemoteDrivePanel, RemoteStatusBar, WinConfiguration->ScpCommander.RemotePanel);  FPanelsRestored = true;  // just to make sure  LocalDirView->DirColProperties->ExtVisible = false;  RemoteDirView->UnixColProperties->ExtVisible = false;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::StorePanelParams(  TCustomDirView * DirView, TControl * DriveControl, TTBXStatusBar * StatusBar,  TScpCommanderPanelConfiguration & PanelConfiguration){  PanelConfiguration.DirViewParams = DirView->ColProperties->ParamsStr;  PanelConfiguration.StatusBar = StatusBar->Visible;  PanelConfiguration.DriveView = DriveControl->Visible;  if (DriveControl->Align == alTop)  {    PanelConfiguration.DriveViewHeight = DriveControl->Height;    PanelConfiguration.DriveViewHeightPixelsPerInch = GetControlPixelsPerInch(this);  }  else  {    PanelConfiguration.DriveViewWidth = DriveControl->Width;    PanelConfiguration.DriveViewWidthPixelsPerInch = GetControlPixelsPerInch(this);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::StoreParams(){  DebugAssert(WinConfiguration);  WinConfiguration->BeginUpdate();  try  {    SaveCommandLine();    TScpCommanderConfiguration CommanderConfiguration = WinConfiguration->ScpCommander;    CommanderConfiguration.ToolbarsLayout = GetToolbarsLayoutStr();    CommanderConfiguration.ToolbarsButtons = GetToolbarsButtonsStr();    CommanderConfiguration.LocalPanelWidth = LeftPanelWidth;    CommanderConfiguration.SessionsTabs = SessionsPageControl->Visible;    CommanderConfiguration.StatusBar = StatusBar->Visible;    CommanderConfiguration.CurrentPanel = FCurrentSide;    StorePanelParams(LocalDirView, LocalDriveView, LocalStatusBar, CommanderConfiguration.LocalPanel);    StorePanelParams(RemoteDirView, RemoteDrivePanel, RemoteStatusBar, CommanderConfiguration.RemotePanel);    CommanderConfiguration.LocalPanel.LastPath = LocalDirView->Path;    CommanderConfiguration.WindowParams = StoreForm(this);    WinConfiguration->ScpCommander = CommanderConfiguration;    TCustomScpExplorerForm::StoreParams();  }  __finally  {    WinConfiguration->EndUpdate();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdateTerminal(TManagedTerminal * Terminal){  DebugAssert(!IsLocalBrowserMode());  TCustomScpExplorerForm::UpdateTerminal(Terminal);  DebugAssert(LocalDirView != NULL);  SAFE_DESTROY(Terminal->LocalExplorerState);  if (WinConfiguration->PreservePanelState)  {    Terminal->LocalExplorerState = LocalDirView->SaveState();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdateSessionData(TSessionData * Data){  DebugAssert(!IsLocalBrowserMode());  // Keep in sync with TSessionData::CopyStateData  TCustomScpExplorerForm::UpdateSessionData(Data);  DebugAssert(LocalDirView);  Data->LocalDirectory = LocalDirView->PathName;  Data->SynchronizeBrowsing = NonVisualDataModule->SynchronizeBrowsingAction->Checked;}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::InternalDDDownload(UnicodeString & TargetDirectory){  DebugAssert(IsFileControl(FDDTargetControl, osLocal));  DebugAssert(!IsLocalBrowserMode());  bool Result = false;  if (FDDTargetControl == LocalDirView)  {    if (LocalDirView->DropTarget)    {      // when drop target is not directory, it is probably file type, which have      // associated drop handler (such as ZIP file in WinXP). in this case we      // must leave drop handling to destination application.      // ! this check is duplicated in LocalDirViewDDTargetHasDropHandler()      // for shellex downloads      if (LocalDirView->ItemIsDirectory(LocalDirView->DropTarget))      {        TargetDirectory = LocalDirView->ItemFullFileName(LocalDirView->DropTarget);        Result = true;      }    }    else    {      TargetDirectory = DefaultDownloadTargetDirectory();      Result = true;    }  }  else if (FDDTargetControl == LocalDriveView)  {    DebugAssert(LocalDriveView->DropTarget != NULL);    TargetDirectory = LocalDriveView->NodePathName(LocalDriveView->DropTarget);    Result = true;  }  else  {    DebugFail();    Abort();  }  return Result;}//---------------------------------------------------------------------------UnicodeString __fastcall TScpCommanderForm::DefaultDownloadTargetDirectory(){  return IncludeTrailingBackslash(LocalDirView->Path);}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::CopyParamDialog(TTransferDirection Direction,  TTransferType Type, bool Temp, TStrings * FileList, UnicodeString & TargetDirectory,  TGUICopyParamType & CopyParam, bool Confirm, bool DragDrop, int Options){  bool Result = false;  // Temp means d&d here so far, may change in future!  if (Temp && (Direction == tdToLocal) &&      IsFileControl(FDDTargetControl, osLocal))  {    Result = InternalDDDownload(TargetDirectory);    if (Result)    {      DebugAssert(FileList->Count > 0);      FInternalDDDownloadList->Assign(FileList);    }  }  else if (!Temp && TargetDirectory.IsEmpty())  {    if (Direction == tdToLocal)    {      TargetDirectory = DefaultDownloadTargetDirectory();    }    else    {      TargetDirectory = UnixIncludeTrailingBackslash(RemoteDirView->Path);    }  }  if (!Result)  {    Result = TCustomScpExplorerForm::CopyParamDialog(Direction, Type, Temp,      FileList, TargetDirectory, CopyParam, Confirm, DragDrop, Options);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DoShow(){  // Make sure the RemoteDirView is disabled (if not connected yet)  // before the focusing below,  // otherwise we disable the view while setting it focused  // (UpdateControls gets called within the SetFocus),  // leading to VCL focus inconsistency with Windows,  // and the view [anything actually] not getting focused after the session  // is finally connected  UpdateControls();  // If we do not call SetFocus on any control before DoShow,  // no control will get focused on Login dialog  // (HACK seems like a bug in VCL)  if (WinConfiguration->ScpCommander.CurrentPanel == osLocal)  {    LocalDirView->SetFocus();    FRemoteDirViewWasFocused = false;  }  else  {    if (DirView(osOther)->Enabled)    {      DirView(osOther)->SetFocus();    }    else    {      LocalDirView->SetFocus();    }    FRemoteDirViewWasFocused = true;  }  TCustomScpExplorerForm::DoShow();}//---------------------------------------------------------------------------Boolean __fastcall TScpCommanderForm::AllowedAction(TAction * Action, TActionAllowed Allowed){  #define FLAG ((TActionFlag)(Action->Tag))  return    TCustomScpExplorerForm::AllowedAction(Action, Allowed) &&    // always require Commander flag    (FLAG & afCommander) &&    // if action is execution or update, we don't require any other flag    // if we check for shortcut, we require proper dirview to be selected    ((Allowed != aaShortCut) ||     ((FLAG & afLocal) && (FCurrentSide == osLocal)) ||     ((FLAG & afRemote) && (FCurrentSide == osRemote))    );  #undef FLAG}//---------------------------------------------------------------------------TCustomDirView * __fastcall TScpCommanderForm::DirView(TOperationSide Side){  Side = GetSide(Side);  if (Side == osLocal)  {    return LocalDirView;  }  else  {    return TCustomScpExplorerForm::DirView(Side);  }}//---------------------------------------------------------------------------TCustomDriveView * __fastcall TScpCommanderForm::DriveView(TOperationSide Side){  Side = GetSide(Side);  if (Side == osLocal)  {    return LocalDriveView;  }  else  {    return TCustomScpExplorerForm::DriveView(Side);  }}//---------------------------------------------------------------------------bool TScpCommanderForm::IsSideLocalBrowser(TOperationSide Side){  return (GetSide(Side) == osLocal);}//---------------------------------------------------------------------------TCustomDirView * TScpCommanderForm::GetCurrentLocalBrowser(){  TOperationSide Side = GetSide(osCurrent);  if (Side == osLocal)  {    return LocalDirView;  }  else  {    DebugFail();    Abort();    return NULL;  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::DirViewEnabled(TOperationSide Side){  Side = GetSide(Side);  if (Side == osLocal)  {    return true;  }  else  {    return TCustomScpExplorerForm::DirViewEnabled(Side);  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::IsFileControl(TObject * Control,  TOperationSide Side){  return    ((Side == osLocal) &&     ((Control == LocalDirView) || (Control == LocalDriveView))) ||    TCustomScpExplorerForm::IsFileControl(Control, Side);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ReloadLocalDirectory(const UnicodeString Directory){  if (Directory.IsEmpty() || SamePaths(Directory, LocalDirView->Path))  {    LocalDirView->ReloadDirectory();    LocalDriveView->ValidateDirectory(LocalDriveView->Selected);  }  TCustomScpExplorerForm::ReloadLocalDirectory();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::BatchStart(void *& Storage){  Storage = new bool;  *static_cast<bool*>(Storage) = LocalDirView->WatchForChanges;  LocalDirView->WatchForChanges = false;  LocalDriveView->WatchDirectory = false;  TCustomScpExplorerForm::BatchStart(Storage);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::BatchEnd(void * Storage){  TCustomScpExplorerForm::BatchEnd(Storage);  DebugAssert(Storage != NULL);  LocalDirView->WatchForChanges = *static_cast<bool*>(Storage);  LocalDriveView->WatchDirectory = LocalDirView->WatchForChanges;  delete Storage;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::StartingDisconnected(){  TCustomScpExplorerForm::StartingDisconnected();  LocalDefaultDirectory();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::TerminalChanged(bool Replaced){  DebugAssert(!IsLocalBrowserMode());  NonVisualDataModule->SynchronizeBrowsingAction->Checked = false;  TCustomScpExplorerForm::TerminalChanged(Replaced);  if (Terminal)  {    // we will load completelly different directory, so particularly    // do not attempt to select previously selected directory    LocalDirView->ContinueSession(false);    // reset home directory    LocalDirView->HomeDirectory = L"";    if (FFirstTerminal || !WinConfiguration->ScpCommander.PreserveLocalDirectory)    {      UnicodeString LocalDirectory = Terminal->StateData->LocalDirectory;      if (!LocalDirectory.IsEmpty())      {        try        {          LocalDirView->Path = LocalDirectory;        }        catch(Exception & E)        {          if (!Terminal->SessionData->UpdateDirectories)          {            Terminal->ShowExtendedException(&E);          }        }      }    }    FFirstTerminal = false;    // Happens when opening a connection from a command-line (StartingDisconnected was not called),    // which does not have a local directory set yet.    if (LocalDirView->Path.IsEmpty())    {      LocalDefaultDirectory();    }    if (WinConfiguration->DefaultDirIsHome &&        !Terminal->SessionData->UpdateDirectories)    {      LocalDirView->HomeDirectory = Terminal->SessionData->LocalDirectoryExpanded;    }    if (WinConfiguration->PreservePanelState &&        !WinConfiguration->ScpCommander.PreserveLocalDirectory)    {      if (Terminal->LocalExplorerState != NULL)      {        LocalDirView->RestoreState(Terminal->LocalExplorerState);      }      else      {        LocalDirView->ClearState();      }      NonVisualDataModule->SynchronizeBrowsingAction->Checked = Terminal->StateData->SynchronizeBrowsing;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDefaultDirectory(){  bool DocumentsDir = true;  UnicodeString LastPath = WinConfiguration->ScpCommander.LocalPanel.LastPath;  if (!LastPath.IsEmpty())  {    try    {      LocalDirView->Path = LastPath;      DocumentsDir = false;    }    catch (...)    {    }  }  if (DocumentsDir)  {    try    {      LocalDirView->HomeDirectory = L"";      UnicodeString HomeDrive = DriveInfo->GetDriveKey(LocalDirView->HomeDirectory);      if (DriveInfo->Get(HomeDrive)->DriveType == DRIVE_REMOTE)      {        LocalDirView->Path = DriveInfo->AnyValidPath();      }      else      {        LocalDirView->ExecuteHomeDirectory();      }    }    catch(Exception & E)    {      ShowExtendedException(NULL, &E);      LocalDirView->Path = ExtractFilePath(Application->ExeName);    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ConfigurationChanged(){  TCustomScpExplorerForm::ConfigurationChanged();  if (WinConfiguration->DefaultDirIsHome && Terminal &&      !Terminal->SessionData->UpdateDirectories)  {    LocalDirView->HomeDirectory = Terminal->SessionData->LocalDirectoryExpanded;  }  else  {    LocalDirView->HomeDirectory = L"";  }  LocalDirView->DimmHiddenFiles = WinConfiguration->DimmHiddenFiles;  LocalDriveView->DimmHiddenDirs = WinConfiguration->DimmHiddenFiles;  LocalDirView->ShowHiddenFiles = WinConfiguration->ShowHiddenFiles;  LocalDirView->FormatSizeBytes = WinConfiguration->FormatSizeBytes;  LocalDriveView->ShowHiddenDirs = WinConfiguration->ShowHiddenFiles;  LocalDirView->ConfirmOverwrite = WinConfiguration->ConfirmOverwriting;  LocalDriveView->ConfirmOverwrite = WinConfiguration->ConfirmOverwriting;  LocalDirView->NortonLike = WinConfiguration->ScpCommander.NortonLikeMode;  RemoteDirView->NortonLike = WinConfiguration->ScpCommander.NortonLikeMode;  LocalDirView->NaturalOrderNumericalSorting = WinConfiguration->NaturalOrderNumericalSorting;  LocalDriveView->NaturalOrderNumericalSorting = WinConfiguration->NaturalOrderNumericalSorting;  LocalDirView->TimeoutShellIconRetrieval = WinConfiguration->TimeoutShellIconRetrieval;  LocalDirView->UseIconUpdateThread = WinConfiguration->UseIconUpdateThread;  if (LocalDirView->RowSelect != WinConfiguration->FullRowSelect)  {    LocalDirView->RowSelect = WinConfiguration->FullRowSelect;    // selection is not redrawn automatically when RowSelect changes    LocalDirView->Invalidate();  }  // See also LocalDirViewDDTargetHasDropHandler  LocalDirView->DragDropFilesEx->ShellExtensions->DropHandler = !WinConfiguration->DDFakeFile;  LocalDriveView->DragDropFilesEx->ShellExtensions->DropHandler = !WinConfiguration->DDFakeFile;  if (Panel(true)->Left > Panel(false)->Left)  {    DisableAlign();    try    {      int AWidth = ClientWidth;      Panel(false)->Align = alClient;      // In bidi mode it gets swapped to alRight, what does not work with the rest of this logic      Splitter->Align = alLeft;      Panel(true)->Align = alLeft;      TControl * ControlsOrder[] =        { Panel(true), Splitter, Panel(false) };      SetHorizontalControlsOrder(ControlsOrder, LENOF(ControlsOrder));      Panel(true)->TabOrder = 0;      Panel(false)->TabOrder = 1;      ClientWidth = AWidth;      LeftPanelWidth = FLastLeftPanelWidth;    }    __finally    {      EnableAlign();    }    int LocalIndex = MenuToolbar->Items->IndexOf(LocalMenuButton);    int RemoteIndex = MenuToolbar->Items->IndexOf(RemoteMenuButton);    MenuToolbar->Items->Move(LocalIndex, RemoteIndex);    RemoteIndex = MenuToolbar->Items->IndexOf(RemoteMenuButton);    MenuToolbar->Items->Move(RemoteIndex, LocalIndex);    SWAP(TShortCut, NonVisualDataModule->LocalChangePathAction->ShortCut,      NonVisualDataModule->RemoteChangePathAction->ShortCut);  }  if ((RemoteDrivePanel->Align == alLeft) != WinConfiguration->ScpCommander.TreeOnLeft)  {    TScpCommanderPanelConfiguration LocalPanel = WinConfiguration->ScpCommander.LocalPanel;    TScpCommanderPanelConfiguration RemotePanel = WinConfiguration->ScpCommander.RemotePanel;    bool TreeOnLeft = WinConfiguration->ScpCommander.TreeOnLeft;    // save value only if it was set yet    if (FPanelsRestored)    {      if (TreeOnLeft)      {        // want to be on left, so it is on top, saving height        LocalPanel.DriveViewHeight = LocalDriveView->Height;        RemotePanel.DriveViewHeight = RemoteDrivePanel->Height;      }      else      {        LocalPanel.DriveViewWidth = LocalDriveView->Width;        RemotePanel.DriveViewWidth = RemoteDrivePanel->Width;      }    }    TAlign NonClientAlign = (TreeOnLeft ? alLeft : alTop);    // VCL adjusts cursors only between crVSplit and crHSplit,    // See TSplitter.RequestAlign    TCursor SplitterCursor = (TreeOnLeft ? crSizeWE : crSizeNS);    LocalDriveView->Align = NonClientAlign;    LocalPanelSplitter->Align = NonClientAlign;    LocalPanelSplitter->Cursor = SplitterCursor;    RemoteDrivePanel->Align = NonClientAlign;    RemotePanelSplitter->Align = NonClientAlign;    RemotePanelSplitter->Cursor = SplitterCursor;    FixControlsPlacement();    if (TreeOnLeft)    {      LocalDriveView->Width = LocalPanel.DriveViewWidth;      RemoteDrivePanel->Width = RemotePanel.DriveViewWidth;    }    else    {      LocalDriveView->Height = LocalPanel.DriveViewHeight;      RemoteDrivePanel->Height = RemotePanel.DriveViewHeight;    }    // in case it trigges config-changed event (does not),    // make sure it does only after we apply the TreeOnLeft change to avoid endless recursion    WinConfiguration->ScpCommander.LocalPanel = LocalPanel;    WinConfiguration->ScpCommander.RemotePanel = RemotePanel;  }  if (FExplorerKeyboardShortcuts != WinConfiguration->ScpCommander.ExplorerKeyboardShortcuts)  {    SetShortcuts();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SetShortcuts(){  // set common norton shorcuts to our actions  NonVisualDataModule->CommanderShortcuts();  FExplorerKeyboardShortcuts = WinConfiguration->ScpCommander.ExplorerKeyboardShortcuts;}//---------------------------------------------------------------------------TPanel * __fastcall TScpCommanderForm::Panel(bool Left){  bool SwappedPanels = WinConfiguration->ScpCommander.SwappedPanels;  if (IsRightToLeft())  {    SwappedPanels = !SwappedPanels;  }  return (SwappedPanels == Left ? RemotePanel : LocalPanel);}//---------------------------------------------------------------------------TPanel * __fastcall TScpCommanderForm::CurrentPanel(){  return (FCurrentSide == osLocal ? LocalPanel : RemotePanel);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SetLeftPanelWidth(double value){  double Total = LocalPanel->Width + RemotePanel->Width;  FLeftPanelWidth = value;  if (value * Total != Panel(true)->Width)  {    Panel(true)->Width = static_cast<int>(value * Total);    UpdateControls();  }}//---------------------------------------------------------------------------double __fastcall TScpCommanderForm::GetLeftPanelWidth(){  return FLeftPanelWidth;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SplitterMoved(TObject * /*Sender*/){  double Left = Panel(true)->Width;  double Total = LocalPanel->Width + RemotePanel->Width;  FLeftPanelWidth = Left / Total;  FLastLeftPanelWidth = LeftPanelWidth;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SplitterCanResize(TObject * /*Sender*/,      int &NewSize, bool & /*Accept*/){  // When splitter is drag so far to right, that width contraint of remote panel would  // be violated, it doesn't stop, but extend form width.  // Following prevents this behavior.  if (ClientWidth - NewSize - Splitter->Width < Panel(false)->Constraints->MinWidth)    NewSize = (ClientWidth - Panel(false)->Constraints->MinWidth - Splitter->Width);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SplitterDblClick(TObject * /*Sender*/){  LeftPanelWidth = 0.5;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::PanelSplitterDblClick(TObject * Sender){  TSplitter * Splitter = dynamic_cast<TSplitter *>(Sender);  DebugAssert(Splitter != NULL);  TControl * DriveView;  TControl * OtherDriveView;  if (Splitter == LocalPanelSplitter)  {    DriveView = LocalDriveView;    OtherDriveView = RemoteDrivePanel;  }  else  {    DriveView = RemoteDrivePanel;    OtherDriveView = LocalDriveView;  }  bool TreeOnLeft = WinConfiguration->ScpCommander.TreeOnLeft;  DebugAssert(DriveView->Visible);  if (OtherDriveView->Visible)  {    if (TreeOnLeft)    {      DriveView->Width = OtherDriveView->Width;    }    else    {      DriveView->Height = OtherDriveView->Height;    }  }  else  {    if (TreeOnLeft)    {      OtherDriveView->Width = DriveView->Width;    }    else    {      OtherDriveView->Height = DriveView->Height;    }    OtherDriveView->Visible = true;  }  FixControlsPlacement();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdateControls(){  TCustomScpExplorerForm::UpdateControls();  UnicodeString SplitterLongHint = Splitter->Hint;  int P = SplitterLongHint.Pos(L"|");  if (P == 0)  {    P = SplitterLongHint.Pos(L"\n");  }  SplitterLongHint.Delete(1, P);  Splitter->Hint = FORMAT(L"%0.0f%%\n%s", (LeftPanelWidth*100, SplitterLongHint));  UnicodeString ACommandLinePromptLabel = LoadStr(COMMAND_LINE_LABEL) + " " +    (!IsSideLocalBrowser(FCurrentSide) ? L"$" : L">");  if (CommandLinePromptLabel->Caption != ACommandLinePromptLabel)  {    CommandLinePromptLabel->Caption = ACommandLinePromptLabel;    // command line combo width needs to be updated as caption width has probably changed    ToolBarResize(CommandLineToolbar);  }  LocalDirView->DarkMode = WinConfiguration->UseDarkTheme();  LocalDriveView->DarkMode = LocalDirView->DarkMode;  LocalDirView->Color = PanelColor();  LocalDriveView->Color = LocalDirView->Color;  LocalDirView->Font->Color = GetWindowTextColor(LocalDirView->Color);  LocalDriveView->Font->Color = LocalDirView->Font->Color;  // TODO  bool LocalSide = (FCurrentSide == osLocal);  TAction * CurrentCopyAction = LocalSide ? NonVisualDataModule->LocalCopyAction : NonVisualDataModule->RemoteCopyAction;  if (CurrentCopyItem->Action != CurrentCopyAction)  {    CurrentCopyItem->Action = CurrentCopyAction;    CurrentCopyToolbar2Item->Action = CurrentCopyAction;    UpdateToolbar2ItemCaption(CurrentCopyToolbar2Item);    CurrentCopyNonQueueItem->Action = LocalSide ? NonVisualDataModule->LocalCopyNonQueueAction : NonVisualDataModule->RemoteCopyNonQueueAction;    CurrentCopyQueueItem->Action = LocalSide ? NonVisualDataModule->LocalCopyQueueAction : NonVisualDataModule->RemoteCopyQueueAction;    TAction * CurrentMoveAction = LocalSide ? NonVisualDataModule->LocalMoveAction : NonVisualDataModule->RemoteMoveAction;    CurrentMoveItem->Action = CurrentMoveAction;    CurrentMoveToolbar2Item->Action = CurrentMoveAction;    UpdateToolbar2ItemCaption(CurrentMoveToolbar2Item);  }  CommandLineCombo->Enabled = IsSideLocalBrowser(FCurrentSide) || CanConsole();  CommandLinePromptLabel->Enabled = CommandLineCombo->Enabled;  // Ad hoc hack to disable the drop down menu, when all its items (and all other buttons on the toolbar) are disabled,  // otherwise it shines too much on the toolbar.  RemoteNewSubmenuItem->Enabled = DirViewEnabled(osRemote);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ChangePath(TOperationSide Side){  DebugAssert((Side == osLocal) || (Side == osRemote));  TTBXComboBoxItem * PathComboBox;  // TODO  if (Side == osLocal)  {    PathComboBox = LocalPathComboBox;  }  else  {    PathComboBox = RemotePathComboBox;  }  ClickToolbarItem(PathComboBox, false);}//---------------------------------------------------------------------------TControl * __fastcall TScpCommanderForm::GetComponent(Byte Component){  switch (Component) {    case fcToolBar2: return Toolbar2Toolbar; // name changed to enforce change of default visibility    case fcStatusBar: return StatusBar;    case fcLocalStatusBar: return LocalStatusBar;    case fcRemoteStatusBar: return RemoteStatusBar;    case fcCommandLinePanel: return CommandLineToolbar;    case fcLocalTree: return LocalDriveView;    case fcSessionToolbar: return SessionToolbar;    case fcCustomCommandsBand: return CustomCommandsToolbar;    case fcColorMenu: return reinterpret_cast<TControl*>(ColorMenuItem);    case fcTransferDropDown: return reinterpret_cast<TControl*>(TransferDropDown);    case fcTransferList: return reinterpret_cast<TControl*>(TransferList);    case fcTransferLabel: return reinterpret_cast<TControl*>(TransferLabel);    case fcLocalPopup: return reinterpret_cast<TControl *>(NonVisualDataModule->LocalFilePopup);    case fcRemotePathComboBox: return reinterpret_cast<TControl*>(RemotePathComboBox);    case fcCommanderMenuBand: return MenuToolbar;    case fcCommanderSessionBand: return SessionToolbar;    case fcCommanderPreferencesBand: return PreferencesToolbar;    case fcCommanderSortBand: return SortToolbar;    case fcCommanderCommandsBand: return CommandsToolbar;    case fcCommanderUpdatesBand: return UpdatesToolbar;    case fcCommanderTransferBand: return TransferToolbar;    case fcCommanderCustomCommandsBand: return CustomCommandsToolbar;    case fcCommanderLocalHistoryBand: return LocalHistoryToolbar;    case fcCommanderLocalNavigationBand: return LocalNavigationToolbar;    case fcCommanderLocalFileBand: return LocalFileToolbar;    case fcCommanderLocalSelectionBand: return LocalSelectionToolbar;    case fcCommanderRemoteHistoryBand: return RemoteHistoryToolbar;    case fcCommanderRemoteNavigationBand: return RemoteNavigationToolbar;    case fcCommanderRemoteFileBand: return RemoteFileToolbar;    case fcCommanderRemoteSelectionBand: return RemoteSelectionToolbar;    default: return TCustomScpExplorerForm::GetComponent(Component);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::FixControlsPlacement(){  TCustomScpExplorerForm::FixControlsPlacement();  QueueSeparatorPanel->Visible = QueuePanel->Visible;  LocalPanelSplitter->Visible = LocalDriveView->Visible;  TControl * ControlsOrder[] =    { BottomDock, QueueSeparatorPanel, QueueSplitter, QueuePanel, StatusBar };  SetVerticalControlsOrder(ControlsOrder, LENOF(ControlsOrder));  TControl * LocalControlsOrder[] =    { LocalTopDock, LocalPathLabel, LocalDriveView, LocalPanelSplitter,      LocalDirView, LocalBottomDock, LocalStatusBar };  SetVerticalControlsOrder(LocalControlsOrder, LENOF(LocalControlsOrder));  SetHorizontalControlsOrder(LocalControlsOrder, LENOF(LocalControlsOrder));  TControl * RemoteControlsOrder[] =    { RemoteTopDock, RemotePathLabel, RemoteDrivePanel, RemotePanelSplitter,      RemoteDirPanel, RemoteBottomDock, RemoteStatusBar };  SetVerticalControlsOrder(RemoteControlsOrder, LENOF(RemoteControlsOrder));  SetHorizontalControlsOrder(RemoteControlsOrder, LENOF(RemoteControlsOrder));  if (LocalDirView->ItemFocused != NULL)  {    LocalDirView->ItemFocused->MakeVisible(false);  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::GetHasDirView(TOperationSide Side){  return TCustomScpExplorerForm::GetHasDirView(Side) || (Side == osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CompareDirectories(){  LocalDirView->CompareFiles(DirView(osOther), false,    WinConfiguration->ScpCommander.CompareCriterias());  DirView(osOther)->CompareFiles(LocalDirView, false,    WinConfiguration->ScpCommander.CompareCriterias());  if (LocalDirView->SelCount + DirView(osOther)->SelCount == 0)  {    UnicodeString Message = MainInstructions(LoadStr(COMPARE_NO_DIFFERENCES));    MessageDialog(Message, qtInformation, qaOK, HELP_COMPARE_NO_DIFFERENCES);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SynchronizeDirectories(){  UnicodeString LocalDirectory = LocalDirView->PathName;  UnicodeString RemoteDirectory = RemoteDirView->PathName;  DoSynchronizeDirectories(LocalDirectory, RemoteDirectory, -1);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::FullSynchronizeDirectories(){  UnicodeString LocalDirectory = LocalDirView->PathName;  UnicodeString RemoteDirectory = RemoteDirView->PathName;  bool SaveMode = !(GUIConfiguration->SynchronizeModeAuto < 0);  TSynchronizeMode Mode =    (SaveMode ? (TSynchronizeMode)GUIConfiguration->SynchronizeModeAuto :      ((FCurrentSide == osLocal) ? smRemote : smLocal));  int Params = GUIConfiguration->SynchronizeParams;  if (DoFullSynchronizeDirectories(LocalDirectory, RemoteDirectory, Mode, Params, SaveMode, -1) >= 0)  {    if (SaveMode)    {      GUIConfiguration->SynchronizeModeAuto = Mode;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ExploreLocalDirectory(){  OpenFolderInExplorer(GetCurrentLocalBrowser()->Path);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewExecFile(TObject *Sender,      TListItem *Item, bool &AllowExec){  DebugAssert(Item);  TDirView * ADirView = DebugNotNull(dynamic_cast<TDirView *>(Sender));  if ((UpperCase(PFileRec(Item->Data)->FileExt) == L"LNK") &&      DirectoryExists(ApiPath(ResolveFileShortCut(ADirView->ItemFullFileName(Item), true))))  {    AllowExec = true;  }  else  {    DoDirViewExecFile(Sender, Item, AllowExec);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalFileControlDDDragEnter(TObject *Sender,      IDataObject *DataObj, int grfKeyState, TPoint &Point, int &dwEffect,      bool &Accept){  // LocalDirViewDDDragEnter is duplication of  // TCustomScpExplorerForm::DirViewDDDragEnter, but it differs in  // literal type of 'DataObj' parameter. The actual type is however the same.  FileControlDDDragEnter(Sender, DataObj, grfKeyState, Point, dwEffect, Accept);}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::PanelOperation(TOperationSide Side,  bool DragDrop){  // TODO  return TCustomScpExplorerForm::PanelOperation(Side, DragDrop) ||    (DropSourceControl == LocalDirView);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::FileOperationProgress(  TFileOperationProgressType & ProgressData){  // Heuristic: When operation finishes and DD target is local dir view,  // we suppose that drag&drop download finished, so local dir view should be  // reloaded  // TODO  if (!ProgressData.InProgress && FProgressForm &&      IsFileControl(FDDTargetControl, osLocal) &&      ((ProgressData.Operation == ::foCopy) || (ProgressData.Operation == ::foMove)))  {    ReloadLocalDirectory();  }  TCustomScpExplorerForm::FileOperationProgress(ProgressData);}//---------------------------------------------------------------------------UnicodeString __fastcall TScpCommanderForm::ChangeFilePath(UnicodeString Path, TOperationSide Side){  TGUICopyParamType CopyParams = GUIConfiguration->CurrentCopyParam;  UnicodeString Result;  while (!Path.IsEmpty())  {    int P = Path.Pos(Side == osLocal ? L'\\' : L'/');    if (P > 0)    {      Result += Terminal->ChangeFileName(&CopyParams, Path.SubString(1, P - 1), Side, false) +        (Side == osLocal ? L'/' : L'\\');      Path.Delete(1, P);    }    else    {      Result += Terminal->ChangeFileName(&CopyParams, Path, osLocal, false);      Path = L"";    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CreateRemoteDirectory(const UnicodeString & Path){  UnicodeString Dir = UnixExtractFileDir(Path);  if (!IsUnixRootPath(Dir) && !Terminal->FileExists(Dir))  {    CreateRemoteDirectory(Dir);  }  TRemoteProperties Properties = GUIConfiguration->NewDirectoryProperties;  TCustomScpExplorerForm::CreateRemoteDirectory(Path, Properties);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SynchronizeBrowsingLocal(  UnicodeString PrevPath, UnicodeString & NewPath, bool Create){  Terminal->ExceptionOnFail = true;  TStrings * Paths = new TStringList();  try  {    Paths->Add(IncludeTrailingBackslash(PrevPath));    Paths->Add(IncludeTrailingBackslash(LocalDirView->Path));    UnicodeString CommonPath;    if (ExtractCommonPath(Paths, CommonPath))    {      PrevPath = IncludeTrailingBackslash(PrevPath);      CommonPath = IncludeTrailingBackslash(CommonPath);      NewPath = RemoteDirView->Path;      while (!SamePaths(PrevPath, CommonPath))      {        if (NewPath == UnixExcludeTrailingBackslash(NewPath))        {          Abort();        }        NewPath = UnixExtractFilePath(UnixExcludeTrailingBackslash(NewPath));        PrevPath = ExtractFilePath(ExcludeTrailingBackslash(PrevPath));      }      NewPath = UnixIncludeTrailingBackslash(NewPath) +        ToUnixPath(LocalDirView->Path.SubString(PrevPath.Length() + 1,          LocalDirView->Path.Length() - PrevPath.Length()));    }    else    {      Abort();    }    if (Create)    {      CreateRemoteDirectory(UnixExcludeTrailingBackslash(NewPath));    }    RemoteDirView->Path = NewPath;  }  __finally  {    Terminal->ExceptionOnFail = false;    delete Paths;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CreateLocalDirectory(const UnicodeString & Path){  UnicodeString Dir = ExtractFileDir(Path);  if (!Dir.IsEmpty() && !DirectoryExists(Dir))  {    CreateLocalDirectory(Dir);  }  LocalDirView->CreateDirectory(Path);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SynchronizeBrowsingRemote(  UnicodeString PrevPath, UnicodeString & NewPath, bool Create){  TStrings * Paths = new TStringList();  try  {    Paths->Add(UnixIncludeTrailingBackslash(PrevPath));    Paths->Add(UnixIncludeTrailingBackslash(RemoteDirView->Path));    UnicodeString CommonPath;    if (UnixExtractCommonPath(Paths, CommonPath))    {      PrevPath = UnixIncludeTrailingBackslash(PrevPath);      CommonPath = UnixIncludeTrailingBackslash(CommonPath);      UnicodeString NewLocalPath;      NewPath = ExcludeTrailingBackslash(LocalDirView->Path);      while (!UnixSamePath(PrevPath, CommonPath))      {        NewLocalPath = ExcludeTrailingBackslash(ExtractFileDir(NewPath));        if (NewLocalPath == NewPath)        {          Abort();        }        NewPath = NewLocalPath;        PrevPath = UnixExtractFilePath(UnixExcludeTrailingBackslash(PrevPath));      }      NewPath = IncludeTrailingBackslash(NewPath) +        ChangeFilePath(          RemoteDirView->Path.SubString(PrevPath.Length() + 1,            RemoteDirView->Path.Length() - PrevPath.Length()),          osRemote);    }    else    {      Abort();    }  }  __finally  {    delete Paths;  }  if (Create)  {    CreateLocalDirectory(ExcludeTrailingBackslash(NewPath));  }  LocalDirView->Path = NewPath;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SynchronizeBrowsing(TCustomDirView * ADirView,  UnicodeString PrevPath, UnicodeString & NewPath, bool Create){  if (ADirView == LocalDirView)  {    SynchronizeBrowsingLocal(PrevPath, NewPath, Create);  }  else  {    SynchronizeBrowsingRemote(PrevPath, NewPath, Create);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SynchronizeBrowsing(TCustomDirView * ADirView){  UnicodeString PrevPath;  // cannot assign to UnicodeString before the class is constructed,  // if we do, the value get lost when UnicodeString constructor gets finally called,  // what results in memory leak in the best case  if (FConstructed)  {    PrevPath = FPrevPath[ADirView == LocalDirView];    FPrevPath[ADirView == LocalDirView] = ADirView->Path;  }  if (!FSynchronisingBrowse && NonVisualDataModule->SynchronizeBrowsingAction->Checked &&      !PrevPath.IsEmpty() && PrevPath != ADirView->Path)  {    DebugAssert(!IsLocalBrowserMode());    TValueRestorer<bool> AllowTransferPresetAutoSelectRestorer(FAllowTransferPresetAutoSelect);    FAllowTransferPresetAutoSelect = false;    TValueRestorer<bool> SynchronisingBrowseRestorer(FSynchronisingBrowse);    FSynchronisingBrowse = true;    try    {      UnicodeString NewPath;      bool Error = false;      std::unique_ptr<TStrings> ErrorMoreMessages;      UnicodeString ErrorHelpKeyword;      try      {        SynchronizeBrowsing(ADirView, PrevPath, NewPath, false);      }      // EAbort means that we do not know how to synchronize browsing      // there's no fallback scenario for that      catch(EAbort &)      {        throw;      }      catch(Exception & E)      {        // what does this say?        if (Application->Terminated)        {          throw;        }        else        {          Error = true;          ErrorMoreMessages.reset(ExceptionToMoreMessages(&E));          ErrorHelpKeyword =            MergeHelpKeyword(HELP_SYNC_DIR_BROWSE_ERROR, GetExceptionHelpKeyword(&E));        }      }      // this was moved here out of the above catch clause,      // to avoid deep nesting, what seems to cause some stray access violations      if (Error)      {        if (MoreMessageDialog(FMTLOAD(SYNC_DIR_BROWSE_CREATE2, (NewPath)),              ErrorMoreMessages.get(), qtConfirmation, qaYes | qaNo,              ErrorHelpKeyword) == qaYes)        {          try          {            SynchronizeBrowsing(ADirView, PrevPath, NewPath, true);          }          catch(Exception & E)          {            if (!Application->Terminated)            {              Terminal->ShowExtendedException(&E);            }            throw;          }        }        else        {          NonVisualDataModule->SynchronizeBrowsingAction->Checked = false;        }      }    }    catch(Exception & E)    {      NonVisualDataModule->SynchronizeBrowsingAction->Checked = false;      // what does this say?      if (Application->Terminated)      {        throw;      }      else      {        MessageDialog(LoadStr(SYNC_DIR_BROWSE_ERROR), qtInformation, qaOK,          HELP_SYNC_DIR_BROWSE_ERROR);      }    }    // note the value restorers at the beginning of this block  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DoDirViewLoaded(TCustomDirView * ADirView){  TCustomScpExplorerForm::DoDirViewLoaded(ADirView);  UpdateControls();  SynchronizeBrowsing(ADirView);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::AddEditLink(TOperationSide Side, bool Add){  // TODO  if (GetSide(Side) == osLocal)  {    bool Edit = false;    UnicodeString FileName;    UnicodeString PointTo;    bool SymbolicLink = true;    if (LocalDirView->ItemFocused)    {      DebugAssert(LocalDirView->ItemFocused->Data);      PFileRec FileRec = (PFileRec)LocalDirView->ItemFocused->Data;      Edit = !Add && (UpperCase(FileRec->FileExt) == L"LNK");      if (Edit)      {        UnicodeString FullName = LocalDirView->ItemFullFileName(LocalDirView->ItemFocused);        FileName = FullName;        PointTo = ResolveFileShortCut(FullName, false);        if (PointTo.IsEmpty())        {          throw Exception(FMTLOAD(RESOLVE_SHORTCUT_ERROR, (FullName)));        }      }      else      {        PointTo = FileRec->FileName;      }    }    if (DoSymlinkDialog(FileName, PointTo, osLocal, SymbolicLink, Edit, false))    {      Configuration->Usage->Inc(L"LocalShortcutsCreated");      DebugAssert(SymbolicLink);      DebugAssert(!FileName.IsEmpty());      DebugAssert(!PointTo.IsEmpty());      if (ExtractFileDrive(FileName) == L"" && FileName[1] != L'\\')      {        FileName = IncludeTrailingBackslash(LocalDirView->PathName) + FileName;      }      if (ExtractFileDrive(PointTo) == L"" && PointTo[1] != L'\\')      {        PointTo = IncludeTrailingBackslash(LocalDirView->PathName) + PointTo;      }      if (ExtractFileExt(FileName) == L"")      {        FileName = FileName + L".lnk";      }      if (Edit)      {        DeleteFileChecked(FileName);      }      if (!CreateFileShortCut(PointTo, FileName, L""))      {        throw Exception(CREATE_SHORTCUT_ERROR);      }    }  }  else  {    TCustomScpExplorerForm::AddEditLink(Side, Add);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DoOpenDirectoryDialog(TOpenDirectoryMode Mode,  TOperationSide Side){  bool UseLocationProfiles;  do  {    UseLocationProfiles = WinConfiguration->UseLocationProfiles;    // Location profiles dialog is not ready to be run without terminal    if (UseLocationProfiles && (Terminal != NULL))    {      TStrings * LocalDirectories = NULL;      TStrings * RemoteDirectories = NULL;      try      {        LocalDirectories = CreateVisitedDirectories(osLocal);        RemoteDirectories = CreateVisitedDirectories(osRemote);        UnicodeString Local = LocalDirView->PathName;        UnicodeString Remote = RemoteDirView->PathName;        if (LocationProfilesDialog(Mode, Side, Local, Remote, LocalDirectories,              RemoteDirectories, Terminal))        {          DoOpenBookmark(Local, Remote);        }      }      __finally      {        delete LocalDirectories;        delete RemoteDirectories;      }    }    else    {      TSynchronizedBrowsingGuard SynchronizedBrowsingGuard;      TCustomScpExplorerForm::DoOpenDirectoryDialog(Mode, Side);    }    // for second and further rounds, always do browse only    Mode = odBrowse;  }  while (UseLocationProfiles != WinConfiguration->UseLocationProfiles);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DoOpenBookmark(UnicodeString Local, UnicodeString Remote){  TSynchronizedBrowsingGuard SynchronizedBrowsingGuard;  // make sure that whatever path is valid it is opened first and only  // after that an eventual error is reported  try  {    if (!Local.IsEmpty())    {      LocalDirView->Path = Local;    }  }  __finally  {    if (!Remote.IsEmpty())    {      // While we might get here when the session is closed (from location profiles),      // it's not a problem as the Path setter is noop then      RemoteDirView->Path = Remote;    }  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::OpenBookmark(TOperationSide Side, TBookmark * Bookmark){  bool Result;  if (WinConfiguration->UseLocationProfiles)  {    DoOpenBookmark(Bookmark->Local, Bookmark->Remote);    Result = true;  }  else  {    Result = TCustomScpExplorerForm::OpenBookmark(Side, Bookmark);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewDDTargetHasDropHandler(  TObject * /*Sender*/, TListItem * Item, int & /*Effect*/, bool & DropHandler){  // When drop target is not directory, it is probably file type, which have  // associated drop handler (such as EXE file). In this case we  // cannot allow drop when when using shellex,  // as drop handlers are disabled, so drop would error  // (see TShellExtension.DropHandler assignment in ConfigurationChanged),  if (WinConfiguration->DDFakeFile &&      !LocalDirView->ItemIsDirectory(Item))  {    DropHandler = false;  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::DDGetTarget(  UnicodeString & Directory, bool & ForceQueue, UnicodeString & CounterName){  bool Result;  if (!FDDFakeFileTarget.IsEmpty())  {    Directory = FDDFakeFileTarget;    FDDFakeFileTarget = L"";    Result = true;    CounterName = L"DownloadsDragDropInternal";    ForceQueue = false;  }  else  {    Result = TCustomScpExplorerForm::DDGetTarget(Directory, ForceQueue, CounterName);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DDFakeFileInitDrag(TFileList * FileList,  bool & Created){  FDDFakeFileTarget = L"";  TCustomScpExplorerForm::DDFakeFileInitDrag(FileList, Created);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalFileControlDDFileOperation(  TObject * /*Sender*/, int dwEffect, UnicodeString SourcePath,  UnicodeString TargetPath, bool Paste, bool & DoOperation){  if (IsFileControl(DropSourceControl, osRemote))  {    UnicodeString TargetDirectory;    if (InternalDDDownload(TargetDirectory))    {      // See TCustomScpExplorerForm::QueueDDProcessDropped      if (FDDExtMapFile != NULL)      {        FDDFakeFileTarget = TargetDirectory;      }      else      {        DebugAssert(FInternalDDDownloadList->Count > 0);        DebugAssert(dwEffect == DROPEFFECT_COPY || dwEffect == DROPEFFECT_MOVE);        TGUICopyParamType CopyParams = GUIConfiguration->CurrentCopyParam;        TTransferType TransferType = dwEffect == DROPEFFECT_COPY ? ttCopy : ttMove;        int Options =          FLAGMASK(DraggingAllFilesFromDirView(osRemote, FInternalDDDownloadList), coAllFiles);        bool NoConfirmation = Paste ? false : (WinConfiguration->DDTransferConfirmation == asOff);        if (CopyParamDialog(tdToLocal, TransferType,              false, FInternalDDDownloadList, TargetDirectory, CopyParams, NoConfirmation, true, Options))        {          int Params =            (TransferType == ttMove ? cpDelete : 0);          DDDownload(FInternalDDDownloadList, TargetDirectory,            &CopyParams, Params);          Configuration->Usage->Inc(L"DownloadsDragDropInternal");          FInternalDDDownloadList->Clear();        }      }      DoOperation = false;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemoteFileControlDDFileOperationExecuted(  TObject * /*Sender*/, int dwEffect, UnicodeString /*SourcePath*/,  UnicodeString /*TargetPath*/){  if ((dwEffect == DROPEFFECT_MOVE) &&      IsFileControl(DropSourceControl, osLocal))  {    ReloadLocalDirectory();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewEnter(TObject * /*Sender*/){  SideEnter(osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDriveViewEnter(TObject * /*Sender*/){  MakeNextInTabOrder(LocalDirView, LocalDriveView);  SideEnter(osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SideEnter(TOperationSide Side){  if (Visible && (IsSideLocalBrowser(FCurrentSide) != IsSideLocalBrowser(Side)))  {    // this may get called yet before controls are initialized    CommandLineCombo->Strings->Clear();    FCommandLineComboPopulated = false;  }  TCustomScpExplorerForm::SideEnter(Side);  if (Visible)  {    UpdateControls();    UpdatePanelsPathLabelsStatus();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdatePanelsPathLabelsStatus(){  LocalPathLabel->UpdateStatus();  RemotePathLabel->UpdateStatus();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::OpenConsole(UnicodeString Command){  SaveCommandLine();  try  {    TCustomScpExplorerForm::OpenConsole(Command);  }  __finally  {    FCommandLineComboPopulated = false;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SaveCommandLine(){  if (FCommandLineComboPopulated)  {    CustomWinConfiguration->History[      !IsSideLocalBrowser(FCurrentSide) ? L"Commands" : L"LocalCommands"] =        CommandLineCombo->Strings;  }}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::ExecuteCommandLine(){  UnicodeString Command = CommandLineCombo->Text;  bool Result =    !NonVisualDataModule->Busy &&    !Command.IsEmpty() &&      (IsSideLocalBrowser(FCurrentSide) ||       (Terminal->AllowedAnyCommand(Command) &&        EnsureCommandSessionFallback(fcAnyCommand)));  if (Result)  {    CommandLinePopulate();    SaveToHistory(CommandLineCombo->Strings, Command);    CommandLineCombo->Text = L"";    if (!IsSideLocalBrowser(FCurrentSide))    {      OpenConsole(Command);    }    else    {      ExecuteShellChecked(Command);    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CommandLinePopulate(){  if (!FCommandLineComboPopulated)  {    TStrings * CommandsHistory;    CommandsHistory = CustomWinConfiguration->History[      !IsSideLocalBrowser(FCurrentSide) ? L"Commands" : L"LocalCommands"];    if ((CommandsHistory != NULL) && (CommandsHistory->Count > 0))    {      CommandLineCombo->Strings = CommandsHistory;    }    else    {      CommandLineCombo->Strings->Clear();    }    FCommandLineComboPopulated = true;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::GoToCommandLine(){  ComponentVisible[fcCommandLinePanel] = true;  if (CommandLineCombo->Enabled)  {    ClickToolbarItem(CommandLineCombo, true);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::GoToTree(){  if (FCurrentSide == osLocal)  {    ComponentVisible[fcLocalTree] = true;    LocalDriveView->SetFocus();  }  else  {    TCustomScpExplorerForm::GoToTree();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::PanelExportStore(TOperationSide Side,  TPanelExport Export, TPanelExportDestination Destination,  TStrings * ExportData){  if (Destination == pedCommandLine)  {    ComponentVisible[fcCommandLinePanel] = true;    UnicodeString Buf;    for (int Index = 0; Index < ExportData->Count; Index++)    {      Buf += ExportData->Strings[Index] + L" ";    }    CommandLineCombo->Text = CommandLineCombo->Text + Buf;  }  else  {    TCustomScpExplorerForm::PanelExportStore(Side, Export, Destination, ExportData);  }}//---------------------------------------------------------------------------int __fastcall TScpCommanderForm::GetStaticComponentsHeight(){  return TCustomScpExplorerForm::GetStaticComponentsHeight() +    (BottomDock->Visible ? BottomDock->Height : 0) +    (QueueSeparatorPanel->Visible ? QueueSeparatorPanel->Height : 0) +    (StatusBar->Visible ? StatusBar->Height : 0);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::SysResizing(unsigned int Cmd){  TCustomScpExplorerForm::SysResizing(Cmd);  if ((Cmd == SC_MAXIMIZE) ||      ((Cmd == SC_DEFAULT) && (WindowState != wsMaximized)))  {    FNormalPanelsWidth = LocalPanel->Width + RemotePanel->Width;  }  else if ((Cmd == SC_RESTORE) ||    ((Cmd == SC_DEFAULT) && (WindowState == wsMaximized)))  {    if (FNormalPanelsWidth >= 0)    {      Panel(true)->Width = static_cast<int>(FLeftPanelWidth * FNormalPanelsWidth);      FNormalPanelsWidth = -1;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::Resize(){  TCustomScpExplorerForm::Resize();  LeftPanelWidth = FLastLeftPanelWidth;  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::PathLabelDblClick(TObject * Sender){  if (!NonVisualDataModule->Busy)  {    OpenDirectory(Sender == LocalPathLabel ? osLocal : osRemote);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathLabelGetStatus(  TCustomPathLabel * /*Sender*/, bool & Active){  // WORKAROUND this strange form is here to make borland compiler work :-)  Active = Active || LocalDriveView->Focused();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemotePathLabelGetStatus(  TCustomPathLabel * /*Sender*/, bool & Active){  // WORKAROUND this strange form is here to make borland compiler work :-)  Active = Active || DriveView(osRemote)->Focused();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathLabelPathClick(  TCustomPathLabel * /*Sender*/, UnicodeString Path){  if (!NonVisualDataModule->Busy)  {    if (SamePaths(Path, LocalDirView->Path))    {      OpenDirectory(osLocal);    }    else    {      LocalDirView->Path = Path;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemotePathLabelPathClick(  TCustomPathLabel * /*Sender*/, UnicodeString Path){  if (!NonVisualDataModule->Busy)  {    if (UnixSamePath(Path, DirView(osRemote)->Path))    {      OpenDirectory(osRemote);    }    else    {      DirView(osRemote)->Path = Path;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewFileIconForName(  TObject * /*Sender*/, TListItem * /*Item*/, UnicodeString & FileName){  UnicodeString PartialExt = Configuration->PartialExt;  if (AnsiSameText(ExtractFileExt(FileName), PartialExt))  {    FileName.SetLength(FileName.Length() - PartialExt.Length());  }  if (WinConfiguration->LocalIconsByExt)  {    FileName = ExtractFileName(FileName);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewUpdateStatusBar(  TObject * /*Sender*/, const TStatusFileInfo & FileInfo){  UpdateFileStatusBar(LocalStatusBar, FileInfo, osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemoteDirViewUpdateStatusBar(  TObject * /*Sender*/, const TStatusFileInfo & FileInfo){  UpdateFileStatusBar(RemoteStatusBar, FileInfo, osRemote);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalStatusBarClick(TObject * /*Sender*/){  LocalDirView->SetFocus();}//---------------------------------------------------------------------------UnicodeString __fastcall TScpCommanderForm::PathForCaption(){  UnicodeString Result;  if (IsSideLocalBrowser(FCurrentSide))  {    // for consistency do not show even local path when there is no terminal    if (Terminal != NULL)    {      switch (WinConfiguration->PathInCaption)      {        case picShort:          {            Result = ExtractFileName(GetCurrentLocalBrowser()->PathName);            if (Result.IsEmpty())            {              Result = GetCurrentLocalBrowser()->PathName;            }          }          break;        case picFull:          Result = GetCurrentLocalBrowser()->PathName;          break;      }    }  }  else  {    Result = TCustomScpExplorerForm::PathForCaption();  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::GetTransferPresetAutoSelectData(  TCopyParamRuleData & Data){  TCustomScpExplorerForm::GetTransferPresetAutoSelectData(Data);  Data.LocalDirectory = LocalDirView->PathName;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemoteDirViewPathChange(TCustomDirView * /*Sender*/){  UpdateRemotePathComboBox(false);  ResetIncrementalSearch();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::UpdateImages(){  TCustomScpExplorerForm::UpdateImages();  TImageList * ImageList = ShellImageListForControl(this, ilsSmall);  RemotePathComboBox->Images = ImageList;  RemotePathComboBox->SubMenuImages = ImageList;  LocalPathComboBox->Images = ImageList;  LocalPathComboBox->SubMenuImages = ImageList;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathComboUpdateDrives(){  // TODO  FLocalSpecialPaths = 0;  TStrings* Strings = LocalPathComboBox->Strings;  Strings->BeginUpdate();  try  {    Strings->Clear();    FLocalPathComboBoxPaths->Clear();    Strings->Add(LoadStr(SPECIAL_FOLDER_MY_DOCUMENTS));    FLocalPathComboBoxPaths->AddObject(GetPersonalFolder(),      (TObject *)DriveInfo->SpecialFolder[CSIDL_PERSONAL]->ImageIndex);    FLocalSpecialPaths++;    Strings->Add(LoadStr(SPECIAL_FOLDER_DESKTOP));    FLocalPathComboBoxPaths->AddObject(GetDesktopFolder(),      (TObject *)DriveInfo->SpecialFolder[CSIDL_DESKTOP]->ImageIndex);    FLocalSpecialPaths++;    std::unique_ptr<TStrings> Drives(LocalDriveView->GetDrives());    for (int Index = 0; Index < Drives->Count; Index++)    {      UnicodeString Drive = Drives->Strings[Index];      if (DriveInfo->Get(Drive)->Valid)      {        UnicodeString Caption = DriveInfo->GetPrettyName(Drive);        if (DriveInfo->IsRealDrive(Drive))        {          Caption.Insert(L"&", 0);        }        Strings->Add(Caption);        UnicodeString RootPath = DriveInfo->GetDriveRoot(Drive);        int ImageIndex = DriveInfo->GetImageIndex(Drive);        FLocalPathComboBoxPaths->AddObject(RootPath, reinterpret_cast<TObject *>(ImageIndex));      }    }  }  __finally  {    Strings->EndUpdate();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathComboUpdate(){  // TODO  // this may get called even after destructor finishes  // (e.g. from SetDockAllowDrag invoked [indirectly] from StoreParams)  if (FLocalPathComboBoxPaths != NULL)  {    DebugAssert(FLocalPathComboBoxPaths->Count == LocalPathComboBox->Strings->Count);    int Index = 0;    while ((Index < FLocalPathComboBoxPaths->Count) &&           !SamePaths(FLocalPathComboBoxPaths->Strings[Index],             LocalDirView->Path.SubString(1, FLocalPathComboBoxPaths->Strings[Index].Length())))    {      Index++;    }    // what to do if not?    if (Index < FLocalPathComboBoxPaths->Count)    {      LocalPathComboBox->ItemIndex = Index;    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewPathChange(TCustomDirView * /*Sender*/){  LocalPathComboUpdate();  ResetIncrementalSearch();  if (IsUncPath(LocalDirView->Path))  {    Configuration->Usage->Inc(L"BrowsedUncPath");  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathComboBoxCancel(TObject * /*Sender*/){  LocalPathComboUpdate();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathComboBoxAdjustImageIndex(  TTBXComboBoxItem * /*Sender*/, const UnicodeString AText, int AIndex,  int & ImageIndex){  // this may get called even before constructor starts  // (e.g. from FixControlsPlacement)  if (FLocalPathComboBoxPaths != NULL)  {    DebugAssert(FLocalPathComboBoxPaths->Count == LocalPathComboBox->Strings->Count);    DebugAssert(AIndex < FLocalPathComboBoxPaths->Count);    if (AIndex < 0)    {      AIndex = LocalPathComboBox->ItemIndex;    }    if (AIndex >= 0)    {      ImageIndex = int(FLocalPathComboBoxPaths->Objects[AIndex]);    }  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathComboBoxItemClick(TObject * /*Sender*/){  DebugAssert(FLocalPathComboBoxPaths->Count == LocalPathComboBox->Strings->Count);  DebugAssert((LocalPathComboBox->ItemIndex >= 0) && (LocalPathComboBox->ItemIndex < FLocalPathComboBoxPaths->Count));  UnicodeString Path = FLocalPathComboBoxPaths->Strings[LocalPathComboBox->ItemIndex];  if (LocalPathComboBox->ItemIndex >= FLocalSpecialPaths)  {    LocalDirView->ExecuteDrive(DriveInfo->GetDriveKey(Path));  }  else  {    LocalDirView->Path = Path;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ToolbarItemResize(TTBXCustomDropDownItem * Item, int Width){  TCustomScpExplorerForm::ToolbarItemResize(Item, Width);  if ((Item == LocalPathComboBox) ||      (Item == RemotePathComboBox) ||      (Item == CommandLineCombo))  {    dynamic_cast<TTBXComboBoxItem *>(Item)->MinListWidth = Width - 4;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CommandLineComboPopup(  TTBCustomItem * /*Sender*/, bool /*FromLink*/){  CommandLinePopulate();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CommandLineComboBeginEdit(  TTBEditItem * /*Sender*/, TTBEditItemViewer * /*Viewer*/, TEdit *EditControl){  InstallPathWordBreakProc(EditControl);  FCommandLineComboEdit = EditControl;  FToolbarEditOldWndProc = EditControl->WindowProc;  EditControl->WindowProc = CommandLineComboEditWndProc;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ExitToolbar(){  CurrentPanel()->SetFocus();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CommandLineComboEditWndProc(TMessage & Message){  bool Handled = false;  if (Message.Msg == WM_CHAR)  {    switch (reinterpret_cast<TWMChar &>(Message).CharCode)    {      case VK_ESCAPE:        CommandLineCombo->Text = L"";        ExitToolbar();        Handled = true;        break;      case VK_TAB:        CommandLineCombo->Text = FCommandLineComboEdit->Text;        ExitToolbar();        Handled = true;        break;      case VK_RETURN:        CommandLineCombo->Text = FCommandLineComboEdit->Text;        ExitToolbar();        if (!ExecuteCommandLine())        {          // re-enter command line          // (most probably exited by now as message dialog was shown)          GoToCommandLine();        }        Handled = true;        break;    }  }  if (!Handled)  {    FToolbarEditOldWndProc(Message);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDriveViewRefreshDrives(TObject * /*Sender*/){  LocalPathComboUpdateDrives();  LocalPathComboUpdate();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::HomeDirectory(TOperationSide Side){  bool SynchronizeBrowsing = NonVisualDataModule->SynchronizeBrowsingAction->Checked;  TSynchronizedBrowsingGuard SynchronizedBrowsingGuard;  TCustomScpExplorerForm::HomeDirectory(Side);  if (SynchronizeBrowsing)  {    TCustomScpExplorerForm::HomeDirectory(GetOtherSize(Side));  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::QueueSubmenuItemPopup(  TTBCustomItem * /*Sender*/, bool /*FromLink*/){  NonVisualDataModule->QueueSpeedComboBoxItemUpdate(QueueSpeedComboBoxItem);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DoFocusRemotePath(TTerminal * Terminal, const UnicodeString & Path){  TSynchronizedBrowsingGuard SynchronizedBrowsingGuard;  TCustomScpExplorerForm::DoFocusRemotePath(Terminal, Path);}//---------------------------------------------------------------------------TOperationSide __fastcall TScpCommanderForm::GetOtherSize(TOperationSide Side){  Side = GetSide(Side);  return ReverseOperationSide(Side);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::HistoryGo(TOperationSide Side, int Index){  TOperationSide OtherSide = GetOtherSize(Side);  if (NonVisualDataModule->SynchronizeBrowsingAction->Checked &&      ((Index < 0) ? (-Index < DirView(OtherSide)->BackCount) : (Index < DirView(OtherSide)->ForwardCount)))  {    TSynchronizedBrowsingGuard SynchronizedBrowsingGuard;    TCustomScpExplorerForm::HistoryGo(Side, Index);    TCustomScpExplorerForm::HistoryGo(OtherSide, Index);  }  else  {    TCustomScpExplorerForm::HistoryGo(Side, Index);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DirViewHistoryGo(  TCustomDirView * Sender, int Index, bool & Cancel){  TOperationSide Side = (Sender == DirView(osRemote) ? osRemote : osLocal);  HistoryGo(Side, Index);  Cancel = true;}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::EligibleForImageDisplayMode(TTBCustomItem * Item){  return    TCustomScpExplorerForm::EligibleForImageDisplayMode(Item) &&    ((Item->Parent == NULL) || (Item->Parent->ParentComponent != Toolbar2Toolbar));}//---------------------------------------------------------------------------bool __fastcall TScpCommanderForm::UpdateToolbarDisplayMode(){  bool Result = TCustomScpExplorerForm::UpdateToolbarDisplayMode();  if (Result)  {    // command line combo width needs to be updated as caption visibility has changed    ToolBarResize(CommandLineToolbar);  }  return Result;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::QueueLabelUpdateStatus(){  TCustomScpExplorerForm::QueueLabelUpdateStatus();  // this is here to deactivate panels path labels when moving focus from  // directory tree to queue  UpdatePanelsPathLabelsStatus();}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalDirViewContextPopup(TObject * /*Sender*/,  TPoint & MousePos, bool & Handled){  if (!WinConfiguration->ScpCommander.SystemContextMenu && !FForceSystemContextMenu)  {    DirViewContextPopupDefaultItem(osLocal, NonVisualDataModule->LocalOpenMenuItem, dcaOpen);    DirViewContextPopupDefaultItem(osLocal, NonVisualDataModule->LocalEditMenuItem, dcaEdit);    DirViewContextPopupDefaultItem(osLocal, NonVisualDataModule->LocalCopyMenuItem, dcaCopy);    DirViewContextPopup(osLocal, fcLocalPopup, MousePos);    Handled = true;  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::DisplaySystemContextMenu(){  TAutoFlag Flag(FForceSystemContextMenu);  if ((FLastContextPopupScreenPoint.x >= 0) && (FLastContextPopupScreenPoint.Y >= 0))  {    LocalDirView->DisplayContextMenu(FLastContextPopupScreenPoint);  }  else  {    LocalDirView->DisplayContextMenuInSitu();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalStatusBarPanelClick(TTBXCustomStatusBar * /*Sender*/,  TTBXStatusPanel * Panel){  FileStatusBarPanelClick(Panel, osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemoteStatusBarPanelClick(TTBXCustomStatusBar * /*Sender*/,  TTBXStatusPanel *Panel){  FileStatusBarPanelClick(Panel, osRemote);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::GoToAddress(){  OpenDirectory(GetSide(osCurrent));}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemotePathLabelMaskClick(TObject * /*Sender*/){  Filter(osRemote);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalPathLabelMaskClick(TObject * /*Sender*/){  Filter(osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::LocalOpenDirButtonPopup(TTBCustomItem * /*Sender*/, bool /*FromLink*/){  CreateOpenDirMenu(LocalOpenDirButton, osLocal);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::RemoteOpenDirButtonPopup(TTBCustomItem * /*Sender*/, bool /*FromLink*/){  CreateOpenDirMenu(RemoteOpenDirButton, osRemote);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::CopyFilesToClipboard(TOperationSide Side, bool OnFocused){  if (IsSideLocalBrowser(Side))  {    TInstantOperationVisualizer Visualizer;    dynamic_cast<TDirView *>(DirView(Side))->CopyToClipBoard(OnFocused);  }  else  {    TCustomScpExplorerForm::CopyFilesToClipboard(Side, OnFocused);  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::PasteFromClipBoard(){  if (DoesClipboardContainOurFiles() && (IsSideLocalBrowser(osCurrent)))  {    if (DebugAlwaysTrue(CanPasteToDirViewFromClipBoard()))    {      ClipboardDownload(DirView(osCurrent)->Path, !WinConfiguration->ConfirmTransferring, false);    }  }  else  {    TCustomScpExplorerForm::PasteFromClipBoard();  }}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::FileColorsChanged(){  TCustomScpExplorerForm::FileColorsChanged();  DoFileColorsChanged(LocalDirView);}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::BrowseFile(){  TCustomScpExplorerForm::BrowseFile();  if (LocalDirView->ItemFocused != NULL)  {    LocalDirView->ItemFocused->Selected = true;  }  TScpCommanderConfiguration ScpCommander = WinConfiguration->ScpCommander;  // Select the panel that has the file, with preference on the remote panel  if (RemoteDirView->ItemFocused->Selected)  {    ScpCommander.CurrentPanel = osRemote;  }  else if (LocalDirView->ItemFocused->Selected)  {    ScpCommander.CurrentPanel = osLocal;  }  WinConfiguration->ScpCommander = ScpCommander;}//---------------------------------------------------------------------------void __fastcall TScpCommanderForm::ThemeChanged(){  TCustomScpExplorerForm::ThemeChanged();  LocalDirView->Perform(WM_THEMECHANGED, 0, 0);}//---------------------------------------------------------------------------
 |