| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087 | //---------------------------------------------------------------------#include <vcl.h>#pragma hdrstop#include <Common.h>#include "WinInterface.h"#include "SynchronizeChecklist.h"#include <Terminal.h>#include <TextsWin.h>#include <CoreMain.h>#include <VCLCommon.h>#include <Tools.h>#include <BaseUtils.hpp>#include <Math.hpp>#include <WinConfiguration.h>#include <GUITools.h>//---------------------------------------------------------------------#pragma link "IEListView"#pragma link "NortonLikeListView"#pragma link "PngImageList"#ifndef NO_RESOURCES#pragma resource "*.dfm"#endif//---------------------------------------------------------------------const int ImageColumnIndex = 4;//---------------------------------------------------------------------bool __fastcall DoSynchronizeChecklistDialog(TSynchronizeChecklist * Checklist,  TSynchronizeMode Mode, int Params, const UnicodeString LocalDirectory,  const UnicodeString RemoteDirectory, TCustomCommandMenuEvent OnCustomCommandMenu){  bool Result;  TSynchronizeChecklistDialog * Dialog = new TSynchronizeChecklistDialog(    Application, Mode, Params, LocalDirectory, RemoteDirectory, OnCustomCommandMenu);  try  {    Result = Dialog->Execute(Checklist);  }  __finally  {    delete Dialog;  }  return Result;}//---------------------------------------------------------------------__fastcall TSynchronizeChecklistDialog::TSynchronizeChecklistDialog(  TComponent * AOwner, TSynchronizeMode Mode, int Params,  const UnicodeString LocalDirectory, const UnicodeString RemoteDirectory,  TCustomCommandMenuEvent OnCustomCommandMenu)  : TForm(AOwner){  FFormRestored = false;  FMode = Mode;  FParams = Params;  FLocalDirectory = ExcludeTrailingBackslash(LocalDirectory);  FRemoteDirectory = UnixExcludeTrailingBackslash(RemoteDirectory);  FOnCustomCommandMenu = OnCustomCommandMenu;  UseSystemSettings(this);  UseDesktopFont(ListView);  UseDesktopFont(StatusBar);  FChecklist = NULL;  FChangingItem = NULL;  FChangingItemIgnore = false;  FChangingItemMass = false;  FGeneralHint = StatusBar->Hint;  SelectScaledImageList(ActionImages);  FOrigListViewWindowProc = ListView->WindowProc;  ListView->WindowProc = ListViewWindowProc;  UpdateImages();  CustomCommandsAction->Visible = (FOnCustomCommandMenu != NULL);  // button visibility cannot be bound to action visibility  CustomCommandsButton2->Visible = CustomCommandsAction->Visible;  MenuButton(CustomCommandsButton2);}//---------------------------------------------------------------------__fastcall TSynchronizeChecklistDialog::~TSynchronizeChecklistDialog(){  ListView->WindowProc = FOrigListViewWindowProc;}//---------------------------------------------------------------------bool __fastcall TSynchronizeChecklistDialog::Execute(TSynchronizeChecklist * Checklist){  FChecklist = Checklist;  bool Result = (ShowModal() == DefaultResult(this));  if (Result)  {    for (int Index = 0; Index < ListView->Items->Count; Index++)    {      TListItem * Item = ListView->Items->Item[Index];      const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);      Checklist->Update(ChecklistItem, Item->Checked, GetChecklistItemAction(ChecklistItem));    }    TSynchronizeChecklistConfiguration FormConfiguration =      CustomWinConfiguration->SynchronizeChecklist;    FormConfiguration.ListParams = ListView->ColProperties->ParamsStr;    UnicodeString WindowParams = FormConfiguration.WindowParams;    // if there is no main window, keep previous "custom pos" indication,    bool CustomPos = (StrToIntDef(CutToChar(WindowParams, L';', true), 0) != 0);    if (Application->MainForm != NULL)    {      CustomPos = (Application->MainForm->BoundsRect != BoundsRect);    }    FormConfiguration.WindowParams =      FORMAT(L"%d;%s", ((CustomPos ? 1 : 0), StoreForm(this)));    CustomWinConfiguration->SynchronizeChecklist = FormConfiguration;  }  return Result;}//---------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UpdateControls(){  StatusBar->Invalidate();  bool AllChecked = true;  bool AllUnchecked = true;  bool AnyBoth = false;  bool AnyNonBoth = false;  TListItem * Item = ListView->Selected;  while (Item != NULL)  {    TSynchronizeChecklist::TAction Action = GetChecklistItemAction(GetChecklistItem(Item));    if ((Action == TSynchronizeChecklist::saUploadUpdate) ||        (Action == TSynchronizeChecklist::saDownloadUpdate))    {      AnyBoth = true;    }    else    {      AnyNonBoth = true;    }    if (Item->Checked)    {      AllUnchecked = false;    }    else    {      AllChecked = false;    }    Item = ListView->GetNextItem(Item, sdAll, TItemStates() << isSelected);  }  EnableControl(OkButton, (FChecked[0] > 0));  CheckAction->Enabled = !AllChecked;  UncheckAction->Enabled = !AllUnchecked;  CheckAllAction->Enabled = (FChecked[0] < FTotals[0]);  UncheckAllAction->Enabled = (FChecked[0] > 0);  CustomCommandsAction->Enabled = AnyBoth && !AnyNonBoth;  ReverseAction->Enabled = (ListView->SelCount > 0);  SelectAllAction->Enabled = (ListView->SelCount < ListView->Items->Count);}//---------------------------------------------------------------------------bool __fastcall TSynchronizeChecklistDialog::GetWindowParams(UnicodeString & WindowParams){  WindowParams = CustomWinConfiguration->SynchronizeChecklist.WindowParams;  bool CustomPos = (StrToIntDef(CutToChar(WindowParams, L';', true), 0) != 0);  return CustomPos || (Application->MainForm == NULL);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CreateParams(TCreateParams & Params){  UnicodeString WindowParams;  if (GetWindowParams(WindowParams))  {    // This is only to set correct TForm::Position. Actual bounds are set later after DPI scaling    RestoreForm(WindowParams, this, true);  }  TForm::CreateParams(Params);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::HelpButtonClick(TObject * /*Sender*/){  FormHelp(this);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::AddSubItem(TListItem * Item, int & Index, const UnicodeString & S){  if (Index < Item->SubItems->Count)  {    Item->SubItems->Strings[Index] = S;  }  else  {    DebugAssert(Index == Item->SubItems->Count);    Item->SubItems->Add(S);  }  Index++;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::LoadItem(TListItem * Item){  UnicodeString S;  const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);  if (ChecklistItem->ImageIndex >= 0)  {    Item->ImageIndex = ChecklistItem->ImageIndex;  }  else  {    Item->ImageIndex = FakeFileImageIndex(ChecklistItem->GetFileName(),      FLAGMASK(ChecklistItem->IsDirectory, FILE_ATTRIBUTE_DIRECTORY));  }  S = ChecklistItem->GetFileName();  if (ChecklistItem->IsDirectory)  {    S = IncludeTrailingBackslash(S);  }  Item->Caption = S;  int Index = 0;  TSynchronizeChecklist::TAction Action = GetChecklistItemAction(ChecklistItem);  if (Action == TSynchronizeChecklist::saDeleteRemote)  {    AddSubItem(Item, Index, L"");    AddSubItem(Item, Index, L"");    AddSubItem(Item, Index, L"");  }  else  {    S = ChecklistItem->Local.Directory;    if (AnsiSameText(FLocalDirectory, S.SubString(1, FLocalDirectory.Length())))    {      S[1] = L'.';      S.Delete(2, FLocalDirectory.Length() - 1);    }    else    {      DebugFail();    }    AddSubItem(Item, Index, S);    if (Action == TSynchronizeChecklist::saDownloadNew)    {      AddSubItem(Item, Index, L"");      AddSubItem(Item, Index, L"");    }    else    {      if (ChecklistItem->IsDirectory)      {        AddSubItem(Item, Index, L"");      }      else      {        AddSubItem(Item, Index,          FormatPanelBytes(ChecklistItem->Local.Size, WinConfiguration->FormatSizeBytes));      }      AddSubItem(Item, Index,        UserModificationStr(ChecklistItem->Local.Modification,          ChecklistItem->Local.ModificationFmt));    }  }  AddSubItem(Item, Index, L"");  DebugAssert(Index == ImageColumnIndex);  if (Action == TSynchronizeChecklist::saDeleteLocal)  {    AddSubItem(Item, Index, L"");    AddSubItem(Item, Index, L"");    AddSubItem(Item, Index, L"");  }  else  {    S = ChecklistItem->Remote.Directory;    if (AnsiSameText(FRemoteDirectory, S.SubString(1, FRemoteDirectory.Length())))    {      S[1] = L'.';      S.Delete(2, FRemoteDirectory.Length() - 1);    }    else    {      DebugFail();    }    AddSubItem(Item, Index, S);    if (Action == TSynchronizeChecklist::saUploadNew)    {      AddSubItem(Item, Index, L"");      AddSubItem(Item, Index, L"");    }    else    {      if (ChecklistItem->IsDirectory)      {        AddSubItem(Item, Index, L"");      }      else      {        AddSubItem(Item, Index,          FormatPanelBytes(ChecklistItem->Remote.Size, WinConfiguration->FormatSizeBytes));      }      AddSubItem(Item, Index, UserModificationStr(ChecklistItem->Remote.Modification,        ChecklistItem->Remote.ModificationFmt));    }  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::LoadList(){  memset(&FTotals, 0, sizeof(FTotals));  memset(&FChecked, 0, sizeof(FChecked));  memset(&FCheckedSize, 0, sizeof(FCheckedSize));  FTotals[0] = FChecklist->Count;  ListView->Items->BeginUpdate();  try  {    ListView->Items->Clear();    for (int Index = 0; Index < FChecklist->Count; Index++)    {      const TSynchronizeChecklist::TItem * ChecklistItem =        FChecklist->Item[ListView->Items->Count];      FChangingItemIgnore = true;      try      {        TListItem * Item = ListView->Items->Add();        TSynchronizeChecklist::TAction Action = ChecklistItem->Action;        FActions.insert(std::make_pair(ChecklistItem, Action));        Item->Data = const_cast<TSynchronizeChecklist::TItem *>(ChecklistItem);        Item->Checked = ChecklistItem->Checked;        LoadItem(Item);      }      __finally      {        FChangingItemIgnore = false;      }      int ActionIndex = int(GetChecklistItemAction(ChecklistItem));      FTotals[ActionIndex]++;      if (ChecklistItem->Checked)      {        FChecked[ActionIndex]++;        FChecked[0]++;        __int64 ItemSize = GetItemSize(ChecklistItem);        FCheckedSize[ActionIndex] += ItemSize;        FCheckedSize[0] += ItemSize;      }    }  }  __finally  {    ListView->Items->EndUpdate();  }  ListView->AlphaSort();  UpdateControls();}//---------------------------------------------------------------------------bool __fastcall TSynchronizeChecklistDialog::IsItemSizeIrrelevant(TSynchronizeChecklist::TAction Action){  switch (Action)  {    case TSynchronizeChecklist::saNone:    case TSynchronizeChecklist::saDeleteRemote:    case TSynchronizeChecklist::saDeleteLocal:      return true;    default:      return false;  }}//---------------------------------------------------------------------------__int64 __fastcall TSynchronizeChecklistDialog::GetItemSize(const TSynchronizeChecklist::TItem * Item){  TSynchronizeChecklist::TAction Action = GetChecklistItemAction(Item);  if (IsItemSizeIrrelevant(Action))  {    return 0;  }  else  {    switch (Action)    {      case TSynchronizeChecklist::saUploadNew:      case TSynchronizeChecklist::saUploadUpdate:        return Item->Local.Size;      case TSynchronizeChecklist::saDownloadNew:      case TSynchronizeChecklist::saDownloadUpdate:        return Item->Remote.Size;      default:        DebugFail();        return 0;    }  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::FormShow(TObject * /*Sender*/){  // Moved here from CreateParams (see also TEditorForm::CreateParams), because there it breaks per-monitor DPI.  // For example BoundsRect is matched to the main form too soon, so it gets rescaled later.  // Also it happens before constructor, what causes UseDesktopFont-flagged controls to rescale twice.  // But Position is already set in the CreateParams, as it cannot be set here anymore.  if (!FFormRestored)  {    FFormRestored = True;    UnicodeString WindowParams;    if (GetWindowParams(WindowParams))    {      RestoreForm(WindowParams, this);    }    else    {      BoundsRect = Application->MainForm->BoundsRect;    }  }  ListView->ColProperties->ParamsStr = CustomWinConfiguration->SynchronizeChecklist.ListParams;  LoadList();  UpdateStatusBarSize();}//---------------------------------------------------------------------------TRect __fastcall TSynchronizeChecklistDialog::GetColumnHeaderRect(int Index){  HWND HeaderHandle = ListView_GetHeader(ListView->Handle);  TRect R;  Header_GetItemRect(HeaderHandle, Index, &R);  TScrollInfo ScrollInfo;  ZeroMemory(&ScrollInfo, sizeof(ScrollInfo));  ScrollInfo.cbSize = sizeof(ScrollInfo);  ScrollInfo.fMask = SIF_POS;  GetScrollInfo(ListView->Handle, SB_HORZ, &ScrollInfo);  R.Left -= ScrollInfo.nPos;  R.Right -= ScrollInfo.nPos;  return R;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewWindowProc(TMessage & Message){  if (Message.Msg == CN_NOTIFY)  {    TWMNotify & NotifyMessage = reinterpret_cast<TWMNotify &>(Message);    if (NotifyMessage.NMHdr->code == NM_CUSTOMDRAW)    {      // workaround      // Due to a bug in VCL, OnAdvancedCustomDrawSubItem is not called for any      // other stage except for cdPrePaint. So we must call it ourselves.      TNMLVCustomDraw * CustomDraw =        reinterpret_cast<TNMLVCustomDraw *>(NotifyMessage.NMHdr);      if (FLAGSET(CustomDraw->nmcd.dwDrawStage, CDDS_ITEM) &&          FLAGSET(CustomDraw->nmcd.dwDrawStage, CDDS_SUBITEM) &&          FLAGSET(CustomDraw->nmcd.dwDrawStage, CDDS_ITEMPOSTPAINT) &&          (CustomDraw->iSubItem == ImageColumnIndex) &&          (ActionImages->Width <= ListView->Columns->Items[CustomDraw->iSubItem]->Width))      {        TListItem * Item = ListView->Items->Item[CustomDraw->nmcd.dwItemSpec];        const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);        TRect HeaderR = GetColumnHeaderRect(CustomDraw->iSubItem);        TRect R = Item->DisplayRect(drBounds);        R.Left = HeaderR.Left + (HeaderR.Width() - ActionImages->Width) / 2;        R.Right = HeaderR.Right;        // workaround        // doing this from ListViewAdvancedCustomDraw corrupts list view on Windows 7        ImageList_Draw(reinterpret_cast<HIMAGELIST>(ActionImages->Handle),          int(GetChecklistItemAction(ChecklistItem)), CustomDraw->nmcd.hdc,          R.Left, ((R.Top + R.Bottom - ActionImages->Height) / 2), ILD_TRANSPARENT);      }    }  }  FOrigListViewWindowProc(Message);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewAdvancedCustomDrawSubItem(  TCustomListView * /*Sender*/, TListItem * /*Item*/, int /*SubItem*/,  TCustomDrawState /*State*/, TCustomDrawStage /*Stage*/, bool & /*DefaultDraw*/){  // this is just fake handler that makes the list view request custom draw notification above}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::StatusBarDrawPanel(  TStatusBar * StatusBar, TStatusPanel * Panel, const TRect & Rect){  bool Possible;  int ActionIndex = Panel->Index;  TSynchronizeChecklist::TAction Action = TSynchronizeChecklist::TAction(ActionIndex);  if (FTotals[ActionIndex] > 0)  {    // if direction is overriden to an action that is otherwise not possible    // in given synchronization mode, we still want to show the stats    Possible = true;  }  else  {    switch (Action)    {      case TSynchronizeChecklist::saNone:        Possible = true;        break;      case TSynchronizeChecklist::saUploadNew:        Possible = ((FMode == smRemote) || (FMode == smBoth)) &&          FLAGCLEAR(FParams, TTerminal::spTimestamp);        break;      case TSynchronizeChecklist::saDownloadNew:        Possible = ((FMode == smLocal) || (FMode == smBoth)) &&          FLAGCLEAR(FParams, TTerminal::spTimestamp);        break;      case TSynchronizeChecklist::saUploadUpdate:        Possible =          ((FMode == smRemote) || (FMode == smBoth)) &&          (FLAGCLEAR(FParams, TTerminal::spNotByTime) || FLAGSET(FParams, TTerminal::spBySize));        break;      case TSynchronizeChecklist::saDownloadUpdate:        Possible =          ((FMode == smLocal) || (FMode == smBoth)) &&          (FLAGCLEAR(FParams, TTerminal::spNotByTime) || FLAGSET(FParams, TTerminal::spBySize));        break;      case TSynchronizeChecklist::saDeleteRemote:        Possible = (FMode == smRemote) &&          FLAGCLEAR(FParams, TTerminal::spTimestamp);        break;      case TSynchronizeChecklist::saDeleteLocal:        Possible = (FMode == smLocal) &&          FLAGCLEAR(FParams, TTerminal::spTimestamp);        break;      default:        DebugFail();        Possible = false;        break;    }  }  UnicodeString PanelText;  if (Possible)  {    PanelText = FORMAT(LoadStrPart(SYNCHRONIZE_SELECTED_ACTIONS, 1),      (FormatNumber(FChecked[ActionIndex]),       FormatNumber(FTotals[ActionIndex])));    if ((FChecked[ActionIndex] > 0) &&        ((ActionIndex == 0) || !IsItemSizeIrrelevant(Action)))    {      PanelText += FORMAT(L" (%s)", (FormatBytes(FCheckedSize[ActionIndex])));    }  }  else  {    PanelText = LoadStrPart(SYNCHRONIZE_SELECTED_ACTIONS, 2);  }  int TextHeight = StatusBar->Canvas->TextHeight(PanelText);  int X = Rect.Left + ActionImages->Width + 4;  int Y = (Rect.Top + Rect.Bottom - TextHeight) / 2;  StatusBar->Canvas->TextRect(Rect, X, Y, PanelText);  X = Rect.Left + 1;  Y = ((Rect.Top + Rect.Bottom - ActionImages->Height) / 2);  int ImageIndex = ActionIndex;  ActionImages->Draw(StatusBar->Canvas, X, Y, ImageIndex, Possible);}//---------------------------------------------------------------------------int __fastcall TSynchronizeChecklistDialog::PanelCount(){  // last "panel" is technical  return StatusBar->Panels->Count - 1;}//---------------------------------------------------------------------------int __fastcall TSynchronizeChecklistDialog::PanelAt(int X){  int Result = 0;  while ((X > StatusBar->Panels->Items[Result]->Width) &&    (Result < PanelCount()))  {    X -= StatusBar->Panels->Items[Result]->Width;    Result++;  }  return ((Result < StatusBar->Panels->Count - 1) ? Result : -1);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::StatusBarMouseMove(  TObject * /*Sender*/, TShiftState /*Shift*/, int X, int /*Y*/){  UnicodeString Hint;  int IPanel = PanelAt(X);  if (IPanel >= 0)  {    Hint = StatusBar->Panels->Items[IPanel]->Text;    if (IPanel > 0)    {      Hint = FORMAT(L"%s\n%s", (Hint, FGeneralHint));    }  }  if (Hint != StatusBar->Hint)  {    Application->CancelHint();    StatusBar->Hint = Hint;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewChange(  TObject * /*Sender*/, TListItem * Item, TItemChange Change){  if ((Change == ctState) && (FChangingItem == Item) && (FChangingItem != NULL))  {    if (!FChangingItemIgnore)    {      DebugAssert(Item->Data != NULL);      if ((FChangingItemChecked != Item->Checked) && (Item->Data != NULL))      {        const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);        int ActionIndex = int(GetChecklistItemAction(ChecklistItem));        int Diff = Item->Checked ? 1 : -1;        FChecked[ActionIndex] += Diff;        FChecked[0] += Diff;        __int64 ItemSize = GetItemSize(ChecklistItem);        if (!Item->Checked)        {          ItemSize = -ItemSize;        }        FCheckedSize[ActionIndex] += ItemSize;        FCheckedSize[0] += ItemSize;        if (!FChangingItemMass)        {          UpdateControls();        }      }    }    FChangingItem = NULL;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewChanging(  TObject * /*Sender*/, TListItem * Item, TItemChange Change,  bool & /*AllowChange*/){  if (Change == ctState)  {    FChangingItem = Item;    FChangingItemChecked = Item->Checked;  }  else  {    DebugAssert(FChangingItem == NULL);    FChangingItem = NULL;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CheckAll(bool Check){  FChangingItemMass = true;  try  {    for (int Index = 0; Index < ListView->Items->Count; Index++)    {      ListView->Items->Item[Index]->Checked = Check;    }  }  __finally  {    FChangingItemMass = false;  }  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CheckAllActionExecute(TObject * /*Sender*/){  CheckAll(true);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UncheckAllActionExecute(TObject * /*Sender*/){  CheckAll(false);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::Check(bool Check){  FChangingItemMass = true;  try  {    TListItem * Item = ListView->Selected;    while (Item != NULL)    {      Item->Checked = Check;      Item = ListView->GetNextItem(Item, sdAll, TItemStates() << isSelected);    }  }  __finally  {    FChangingItemMass = false;  }  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CheckActionExecute(TObject * /*Sender*/){  Check(true);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UncheckActionExecute(TObject * /*Sender*/){  Check(false);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewSelectItem(  TObject * /*Sender*/, TListItem * /*Item*/, bool /*Selected*/){  // Delayed update of button status in case many items are being selected at once  // Also change of selection causes buttons to flash, as for short period of time,  // no item is selected  UpdateTimer->Enabled = true;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UpdateTimerTimer(  TObject * /*Sender*/){  UpdateTimer->Enabled = false;  UpdateControls();}//---------------------------------------------------------------------------TListItem * __fastcall TSynchronizeChecklistDialog::SelectAll(bool Select, int Action,  bool OnlyTheAction){  TListItem * Result = NULL;  for (int Index = 0; Index < ListView->Items->Count; Index++)  {    TListItem * Item = ListView->Items->Item[Index];    if (Action == 0)    {      Item->Selected = Select;      if (Result == NULL)      {        Result = Item;      }    }    else    {      const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);      bool WantedAction = (int(GetChecklistItemAction(ChecklistItem)) == Action);      if (WantedAction || !OnlyTheAction)      {        Item->Selected = Select && WantedAction;        if (WantedAction && (Result == NULL))        {          Result = Item;        }      }    }  }  return Result;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::SelectAllActionExecute(  TObject * /*Sender*/){  SelectAll(true);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::StatusBarMouseDown(  TObject * /*Sender*/, TMouseButton /*Button*/, TShiftState Shift, int X,  int /*Y*/){  int IPanel = PanelAt(X);  if (IPanel >= 0)  {    TListItem * Item = SelectAll(true, IPanel, Shift.Contains(ssCtrl));    if (Item != NULL)    {      Item->MakeVisible(false);      Item->Focused = true;      ListView->SetFocus();    }  }}//---------------------------------------------------------------------------int __fastcall TSynchronizeChecklistDialog::CompareNumber(__int64 Value1,  __int64 Value2){  int Result;  if (Value1 < Value2)  {    Result = -1;  }  else if (Value1 == Value2)  {    Result = 0;  }  else  {    Result = 1;  }  return Result;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewCompare(  TObject * /*Sender*/, TListItem * Item1, TListItem * Item2, int /*Data*/,  int & Compare){  const TSynchronizeChecklist::TItem * ChecklistItem1 = GetChecklistItem(Item1);  const TSynchronizeChecklist::TItem * ChecklistItem2 = GetChecklistItem(Item2);  TIEListViewColProperties * ColProperties =    dynamic_cast<TIEListViewColProperties *>(ListView->ColProperties);  switch (ColProperties->SortColumn)  {    case 0: // name      Compare = AnsiCompareText(ChecklistItem1->GetFileName(), ChecklistItem2->GetFileName());      break;    // sorting by local and remote dir is the same    case 1: // local dir    case 5: // remote dir      Compare = 0; // default sorting      break;    case 2: // local size      Compare = CompareNumber(ChecklistItem1->Local.Size, ChecklistItem2->Local.Size);      break;    case 3: // local changed      Compare = CompareFileTime(ChecklistItem1->Local.Modification,        ChecklistItem2->Local.Modification);      break;    case ImageColumnIndex: // action      Compare = CompareNumber(GetChecklistItemAction(ChecklistItem1), GetChecklistItemAction(ChecklistItem2));      break;    case 6: // remote size      Compare = CompareNumber(ChecklistItem1->Remote.Size, ChecklistItem2->Remote.Size);      break;    case 7: // remote changed      Compare = CompareFileTime(ChecklistItem1->Remote.Modification,        ChecklistItem2->Remote.Modification);      break;  }  if (Compare == 0)  {    if (!ChecklistItem1->Local.Directory.IsEmpty())    {      Compare = AnsiCompareText(ChecklistItem1->Local.Directory, ChecklistItem2->Local.Directory);    }    else    {      DebugAssert(!ChecklistItem1->Remote.Directory.IsEmpty());      Compare = AnsiCompareText(ChecklistItem1->Remote.Directory, ChecklistItem2->Remote.Directory);    }    if (Compare == 0)    {      Compare = AnsiCompareText(ChecklistItem1->GetFileName(), ChecklistItem2->GetFileName());    }  }  if (!ColProperties->SortAscending)  {    Compare = -Compare;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewSecondaryColumnHeader(  TCustomIEListView * /*Sender*/, int Index, int & SecondaryColumn){  // "remote dir" column is sorting alias for "local dir" column  if (Index == 5)  {    SecondaryColumn = 1;  }  else  {    SecondaryColumn = -1;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewContextPopup(  TObject * Sender, TPoint & MousePos, bool & Handled){  // to update source popup menu before TBX menu is created  UpdateControls();  MenuPopup(Sender, MousePos, Handled);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CustomCommandsActionExecute(  TObject * /*Sender*/){  TStrings * LocalFileList = new TStringList();  TStrings * RemoteFileList = new TStringList();  try  {    TListItem * Item = ListView->Selected;    DebugAssert(Item != NULL);    while (Item != NULL)    {      const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);      DebugAssert((GetChecklistItemAction(ChecklistItem) == TSynchronizeChecklist::saUploadUpdate) ||             (GetChecklistItemAction(ChecklistItem) == TSynchronizeChecklist::saDownloadUpdate));      DebugAssert(ChecklistItem->RemoteFile != NULL);      UnicodeString LocalPath =        IncludeTrailingBackslash(ChecklistItem->Local.Directory) +        ChecklistItem->Local.FileName;      LocalFileList->Add(LocalPath);      UnicodeString RemotePath =        UnixIncludeTrailingBackslash(ChecklistItem->Remote.Directory) +        ChecklistItem->Remote.FileName;      RemoteFileList->AddObject(RemotePath, ChecklistItem->RemoteFile);      Item = ListView->GetNextItem(Item, sdAll, TItemStates() << isSelected);    }  }  catch(...)  {    delete LocalFileList;    delete RemoteFileList;    throw;  }  DebugAssert(FOnCustomCommandMenu != NULL);  FOnCustomCommandMenu(CustomCommandsAction, LocalFileList, RemoteFileList);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UpdateStatusBarSize(){  int PanelWidth = Min(StatusBar->Width / PanelCount(), ScaleByTextHeight(this, 160));  for (int Index = 0; Index < PanelCount(); Index++)  {    StatusBar->Panels->Items[Index]->Width = PanelWidth;  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::StatusBarResize(TObject * /*Sender*/){  UpdateStatusBarSize();}//---------------------------------------------------------------------------const TSynchronizeChecklist::TItem * TSynchronizeChecklistDialog::GetChecklistItem(  TListItem * Item){  return static_cast<const TSynchronizeChecklist::TItem *>(Item->Data);}//---------------------------------------------------------------------------TSynchronizeChecklist::TAction & TSynchronizeChecklistDialog::GetChecklistItemAction(  const TSynchronizeChecklist::TItem * ChecklistItem){  TActions::iterator i = FActions.find(ChecklistItem);  if (i == FActions.end())  {    throw EInvalidOperation(L"");  }  return i->second;}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ReverseActionExecute(TObject * /*Sender*/){  TAutoFlag Flag(FChangingItemMass);  TListItem * Item = ListView->Selected;  while (Item != NULL)  {    const TSynchronizeChecklist::TItem * ChecklistItem = GetChecklistItem(Item);    TSynchronizeChecklist::TAction & Action = GetChecklistItemAction(ChecklistItem);    TSynchronizeChecklist::TAction NewAction = TSynchronizeChecklist::Reverse(Action);    if (DebugAlwaysTrue(Action != NewAction))    {      int ActionIndex = int(Action);      FTotals[ActionIndex]--;      if (Item->Checked)      {        FChecked[ActionIndex]--;        __int64 ItemSize = GetItemSize(ChecklistItem);        FCheckedSize[ActionIndex] -= ItemSize;        FCheckedSize[0] -= ItemSize;      }      Action = NewAction;      ActionIndex = int(Action);      FTotals[ActionIndex]++;      if (Item->Checked)      {        FChecked[ActionIndex]++;        // item size may differ with action (0 for delete, but non-0 for new file transfer)        __int64 ItemSize = GetItemSize(ChecklistItem);        FCheckedSize[ActionIndex] += ItemSize;        FCheckedSize[0] += ItemSize;      }      LoadItem(Item);    }    Item = ListView->GetNextItem(Item, sdAll, TItemStates() << isSelected);  }  Flag.Release();  UpdateControls();}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::ListViewClick(TObject * /*Sender*/){  TKeyboardState KeyState;  GetKeyboardState(KeyState);  TShiftState ShiftState = KeyboardStateToShiftState(KeyState);  // when selecting, do not reverse, even when user clicked on action column  if (!ShiftState.Contains(ssShift) && !ShiftState.Contains(ssCtrl))  {    TPoint P = ListView->ScreenToClient(Mouse->CursorPos);    TRect R = GetColumnHeaderRect(ImageColumnIndex);    if ((R.Left <= P.x) && (P.x <= R.Right))    {      // If no item was selected before the click, the action is not enabled yet here,      // and Execute would be noop, force update      UpdateControls();      ReverseAction->Execute();    }  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::Dispatch(void * Message){  TMessage * M = reinterpret_cast<TMessage*>(Message);  if (M->Msg == WM_SYSCOMMAND)  {    if (!HandleMinimizeSysCommand(*M))    {      TForm::Dispatch(Message);    }  }  else if (M->Msg == CM_DPICHANGED)  {    CMDpiChanged(*M);  }  else  {    TForm::Dispatch(Message);  }}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::UpdateImages(){  ListView->SmallImages = ShellImageListForControl(this, ilsSmall);}//---------------------------------------------------------------------------void __fastcall TSynchronizeChecklistDialog::CMDpiChanged(TMessage & Message){  TForm::Dispatch(&Message);  UpdateImages();}//---------------------------------------------------------------------------
 |