Synchronize.cpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
  1. //---------------------------------------------------------------------------
  2. #include <vcl.h>
  3. #pragma hdrstop
  4. #include <Common.h>
  5. #include "WinInterface.h"
  6. #include "Synchronize.h"
  7. #include "VCLCommon.h"
  8. #include "CopyParams.h"
  9. #include "Terminal.h"
  10. #include "GUITools.h"
  11. #include <CoreMain.h>
  12. #include <Configuration.h>
  13. #include <TextsWin.h>
  14. #include <HelpWin.h>
  15. #include <CustomWinConfiguration.h>
  16. //---------------------------------------------------------------------------
  17. #pragma package(smart_init)
  18. #pragma link "HistoryComboBox"
  19. #pragma link "GrayedCheckBox"
  20. #ifndef NO_RESOURCES
  21. #pragma resource "*.dfm"
  22. #endif
  23. //---------------------------------------------------------------------------
  24. const int WM_USER_STOP = WM_WINSCP_USER + 2;
  25. //---------------------------------------------------------------------------
  26. bool __fastcall DoSynchronizeDialog(TSynchronizeParamType & Params,
  27. const TCopyParamType * CopyParams, TSynchronizeStartStopEvent OnStartStop,
  28. bool & SaveSettings, int Options, int CopyParamAttrs,
  29. TGetSynchronizeOptionsEvent OnGetOptions, bool Start)
  30. {
  31. bool Result;
  32. TSynchronizeDialog * Dialog = new TSynchronizeDialog(Application,
  33. OnStartStop, OnGetOptions, Start);
  34. try
  35. {
  36. Dialog->Options = Options;
  37. Dialog->CopyParamAttrs = CopyParamAttrs;
  38. Dialog->Params = Params;
  39. Dialog->CopyParams = *CopyParams;
  40. Dialog->SaveSettings = SaveSettings;
  41. Result = Dialog->Execute();
  42. if (Result)
  43. {
  44. SaveSettings = Dialog->SaveSettings;
  45. Params = Dialog->Params;
  46. }
  47. }
  48. __finally
  49. {
  50. delete Dialog;
  51. }
  52. return Result;
  53. }
  54. //---------------------------------------------------------------------------
  55. const TSynchronizeDialog::MaxLogItems = 1000;
  56. //---------------------------------------------------------------------------
  57. __fastcall TSynchronizeDialog::TSynchronizeDialog(TComponent * Owner,
  58. TSynchronizeStartStopEvent OnStartStop, TGetSynchronizeOptionsEvent OnGetOptions,
  59. bool StartImmediatelly)
  60. : TForm(Owner)
  61. {
  62. UseSystemSettings(this);
  63. FOptions = 0;
  64. FSynchronizing = false;
  65. FMinimizedByMe = false;
  66. FPresetsMenu = new TPopupMenu(this);
  67. FOnStartStop = OnStartStop;
  68. FOnGetOptions = OnGetOptions;
  69. FSynchronizeOptions = NULL;
  70. FStartImmediatelly = StartImmediatelly;
  71. HotTrackLabel(CopyParamLabel);
  72. CopyParamListButton(TransferSettingsButton);
  73. SetGlobalMinimizeHandler(this, GlobalMinimize);
  74. }
  75. //---------------------------------------------------------------------------
  76. __fastcall TSynchronizeDialog::~TSynchronizeDialog()
  77. {
  78. // if application is closing OnCloseQuery might not get called
  79. // (this particularly happens if last terminal is disconnected while dialog is
  80. // open)
  81. if (FSynchronizing)
  82. {
  83. OnlyStop();
  84. }
  85. ClearGlobalMinimizeHandler(GlobalMinimize);
  86. delete FSynchronizeOptions;
  87. delete FPresetsMenu;
  88. }
  89. //---------------------------------------------------------------------------
  90. void __fastcall TSynchronizeDialog::UpdateControls()
  91. {
  92. EnableControl(StartButton, !LocalDirectoryEdit->Text.IsEmpty() &&
  93. !RemoteDirectoryEdit->Text.IsEmpty());
  94. TButton * OldButton = FSynchronizing ? StartButton : StopButton;
  95. TButton * NewButton = FSynchronizing ? StopButton : StartButton;
  96. if (!NewButton->Visible || OldButton->Visible)
  97. {
  98. NewButton->Visible = true;
  99. if (OldButton->Focused())
  100. {
  101. NewButton->SetFocus();
  102. }
  103. OldButton->Default = false;
  104. NewButton->Default = true;
  105. OldButton->Visible = false;
  106. // some of the above steps hides accelerators when start button is pressed with mouse
  107. ResetSystemSettings(this);
  108. }
  109. Caption = LoadStr(FSynchronizing ? SYNCHRONIZE_SYCHRONIZING : SYNCHRONIZE_TITLE);
  110. EnableControl(TransferSettingsButton, !FSynchronizing);
  111. CancelButton->Visible = !FSynchronizing || FLAGSET(FOptions, soNoMinimize);
  112. EnableControl(CancelButton, !FSynchronizing);
  113. EnableControl(DirectoriesGroup, !FSynchronizing);
  114. EnableControl(OptionsGroup, !FSynchronizing);
  115. EnableControl(CopyParamGroup, !FSynchronizing);
  116. MinimizeButton->Visible = FSynchronizing && FLAGCLEAR(FOptions, soNoMinimize);
  117. EnableControl(SynchronizeSelectedOnlyCheck,
  118. OptionsGroup->Enabled && FLAGSET(FOptions, soAllowSelectedOnly));
  119. UnicodeString InfoStr = CopyParams.GetInfoStr(L"; ", ActualCopyParamAttrs());
  120. CopyParamLabel->Caption = InfoStr;
  121. CopyParamLabel->Hint = InfoStr;
  122. CopyParamLabel->ShowHint =
  123. (CopyParamLabel->Canvas->TextWidth(InfoStr) > (CopyParamLabel->Width * 3 / 2));
  124. TransferSettingsButton->Style =
  125. FLAGCLEAR(Options, soDoNotUsePresets) ?
  126. TCustomButton::bsSplitButton : TCustomButton::bsPushButton;
  127. if (LogPanel->Visible != FSynchronizing)
  128. {
  129. if (FSynchronizing)
  130. {
  131. LogPanel->Visible = true;
  132. ClientHeight = ClientHeight + LogPanel->Height;
  133. }
  134. else
  135. {
  136. ClientHeight = ClientHeight - LogPanel->Height;
  137. LogPanel->Visible = false;
  138. }
  139. }
  140. }
  141. //---------------------------------------------------------------------------
  142. void __fastcall TSynchronizeDialog::ControlChange(TObject * /*Sender*/)
  143. {
  144. UpdateControls();
  145. }
  146. //---------------------------------------------------------------------------
  147. bool __fastcall TSynchronizeDialog::Execute()
  148. {
  149. // at start assume that copy param is current preset
  150. FPreset = GUIConfiguration->CopyParamCurrent;
  151. LocalDirectoryEdit->Items = CustomWinConfiguration->History[L"LocalDirectory"];
  152. RemoteDirectoryEdit->Items = CustomWinConfiguration->History[L"RemoteDirectory"];
  153. ShowModal();
  154. return true;
  155. }
  156. //---------------------------------------------------------------------------
  157. void __fastcall TSynchronizeDialog::SetParams(const TSynchronizeParamType& value)
  158. {
  159. FParams = value;
  160. RemoteDirectoryEdit->Text = value.RemoteDirectory;
  161. LocalDirectoryEdit->Text = value.LocalDirectory;
  162. SynchronizeDeleteCheck->Checked = FLAGSET(value.Params, spDelete);
  163. SynchronizeExistingOnlyCheck->Checked = FLAGSET(value.Params, spExistingOnly);
  164. SynchronizeSelectedOnlyCheck->Checked = FLAGSET(value.Params, spSelectedOnly);
  165. SynchronizeRecursiveCheck->Checked = FLAGSET(value.Options, soRecurse);
  166. SynchronizeSynchronizeCheck->State =
  167. FLAGSET(value.Options, soSynchronizeAsk) ? cbGrayed :
  168. (FLAGSET(value.Options, soSynchronize) ? cbChecked : cbUnchecked);
  169. }
  170. //---------------------------------------------------------------------------
  171. TSynchronizeParamType __fastcall TSynchronizeDialog::GetParams()
  172. {
  173. TSynchronizeParamType Result = FParams;
  174. Result.RemoteDirectory = RemoteDirectoryEdit->Text;
  175. Result.LocalDirectory = LocalDirectoryEdit->Text;
  176. Result.Params =
  177. (Result.Params & ~(spDelete | spExistingOnly | spSelectedOnly | spTimestamp)) |
  178. FLAGMASK(SynchronizeDeleteCheck->Checked, spDelete) |
  179. FLAGMASK(SynchronizeExistingOnlyCheck->Checked, spExistingOnly) |
  180. FLAGMASK(SynchronizeSelectedOnlyCheck->Checked, spSelectedOnly);
  181. Result.Options =
  182. (Result.Options & ~(soRecurse | soSynchronize | soSynchronizeAsk)) |
  183. FLAGMASK(SynchronizeRecursiveCheck->Checked, soRecurse) |
  184. FLAGMASK(SynchronizeSynchronizeCheck->State == cbChecked, soSynchronize) |
  185. FLAGMASK(SynchronizeSynchronizeCheck->State == cbGrayed, soSynchronizeAsk);
  186. return Result;
  187. }
  188. //---------------------------------------------------------------------------
  189. void __fastcall TSynchronizeDialog::LocalDirectoryBrowseButtonClick(
  190. TObject * /*Sender*/)
  191. {
  192. UnicodeString Directory = LocalDirectoryEdit->Text;
  193. if (SelectDirectory(Directory, LoadStr(SELECT_LOCAL_DIRECTORY), false))
  194. {
  195. LocalDirectoryEdit->Text = Directory;
  196. }
  197. }
  198. //---------------------------------------------------------------------------
  199. void __fastcall TSynchronizeDialog::SetOptions(int value)
  200. {
  201. if (Options != value)
  202. {
  203. FOptions = value;
  204. UpdateControls();
  205. }
  206. }
  207. //---------------------------------------------------------------------------
  208. void __fastcall TSynchronizeDialog::CopyParamListPopup(TRect R, int AdditionalOptions)
  209. {
  210. // We pass in FCopyParams, although it may not be the exact copy param
  211. // that will be used (because of PreserveTime). The reason is to
  212. // display checkbox next to user-selected preset
  213. ::CopyParamListPopup(
  214. R, FPresetsMenu, FCopyParams, FPreset, CopyParamClick,
  215. cplCustomize | AdditionalOptions, ActualCopyParamAttrs());
  216. }
  217. //---------------------------------------------------------------------------
  218. void __fastcall TSynchronizeDialog::TransferSettingsButtonClick(
  219. TObject * /*Sender*/)
  220. {
  221. if (FLAGCLEAR(FOptions, soDoNotUsePresets) && !SupportsSplitButton())
  222. {
  223. CopyParamListPopup(CalculatePopupRect(TransferSettingsButton), 0);
  224. }
  225. else
  226. {
  227. CopyParamGroupClick(NULL);
  228. }
  229. }
  230. //---------------------------------------------------------------------------
  231. void __fastcall TSynchronizeDialog::DoStartStop(bool Start, bool Synchronize)
  232. {
  233. if (FOnStartStop)
  234. {
  235. TSynchronizeParamType SParams = GetParams();
  236. SParams.Options =
  237. (SParams.Options & ~(soSynchronize | soSynchronizeAsk)) |
  238. FLAGMASK(Synchronize, soSynchronize);
  239. if (Start)
  240. {
  241. delete FSynchronizeOptions;
  242. FSynchronizeOptions = new TSynchronizeOptions;
  243. FOnGetOptions(SParams.Params, *FSynchronizeOptions);
  244. }
  245. FOnStartStop(this, Start, SParams, CopyParams, FSynchronizeOptions, DoAbort,
  246. NULL, DoLog);
  247. }
  248. }
  249. //---------------------------------------------------------------------------
  250. void __fastcall TSynchronizeDialog::Dispatch(void * Message)
  251. {
  252. assert(Message);
  253. if ((reinterpret_cast<TMessage *>(Message)->Msg == WM_USER_STOP) && FAbort)
  254. {
  255. if (FSynchronizing)
  256. {
  257. Stop();
  258. }
  259. if (FClose)
  260. {
  261. FClose = false;
  262. ModalResult = mrCancel;
  263. }
  264. }
  265. else
  266. {
  267. TForm::Dispatch(Message);
  268. }
  269. }
  270. //---------------------------------------------------------------------------
  271. void __fastcall TSynchronizeDialog::DoAbort(TObject * /*Sender*/, bool Close)
  272. {
  273. FAbort = true;
  274. FClose = Close;
  275. PostMessage(Handle, WM_USER_STOP, 0, 0);
  276. }
  277. //---------------------------------------------------------------------------
  278. void __fastcall TSynchronizeDialog::DoLog(TSynchronizeController * /*Controller*/,
  279. TSynchronizeLogEntry Entry, const UnicodeString Message)
  280. {
  281. LogView->Items->BeginUpdate();
  282. try
  283. {
  284. TListItem * Item = LogView->Items->Add();
  285. Item->Caption = Now().TimeString();
  286. Item->SubItems->Add(Message);
  287. Item->MakeVisible(false);
  288. while (LogView->Items->Count > MaxLogItems)
  289. {
  290. LogView->Items->Delete(0);
  291. }
  292. }
  293. __finally
  294. {
  295. LogView->Items->EndUpdate();
  296. if (Entry == slScan)
  297. {
  298. // redraw log before the scanning block update
  299. LogView->Repaint();
  300. }
  301. }
  302. }
  303. //---------------------------------------------------------------------------
  304. void __fastcall TSynchronizeDialog::StartButtonClick(TObject * /*Sender*/)
  305. {
  306. bool Synchronize;
  307. bool Continue = true;
  308. if (SynchronizeSynchronizeCheck->State == cbGrayed)
  309. {
  310. TMessageParams Params(mpNeverAskAgainCheck);
  311. switch (MoreMessageDialog(LoadStr(SYNCHRONISE_BEFORE_KEEPUPTODATE2),
  312. NULL, qtConfirmation, qaYes | qaNo | qaCancel, HELP_KEEPUPTODATE_SYNCHRONIZE,
  313. &Params))
  314. {
  315. case qaNeverAskAgain:
  316. SynchronizeSynchronizeCheck->State = cbChecked;
  317. // fall thru
  318. case qaYes:
  319. Synchronize = true;
  320. break;
  321. case qaNo:
  322. Synchronize = false;
  323. break;
  324. default:
  325. case qaCancel:
  326. Continue = false;
  327. break;
  328. };
  329. }
  330. else
  331. {
  332. Synchronize = SynchronizeSynchronizeCheck->Checked;
  333. }
  334. if (Continue)
  335. {
  336. assert(!FSynchronizing);
  337. LocalDirectoryEdit->SaveToHistory();
  338. CustomWinConfiguration->History[L"LocalDirectory"] = LocalDirectoryEdit->Items;
  339. RemoteDirectoryEdit->SaveToHistory();
  340. CustomWinConfiguration->History[L"RemoteDirectory"] = RemoteDirectoryEdit->Items;
  341. FSynchronizing = true;
  342. try
  343. {
  344. UpdateControls();
  345. Repaint();
  346. FAbort = false;
  347. DoStartStop(true, Synchronize);
  348. }
  349. catch(...)
  350. {
  351. FSynchronizing = false;
  352. UpdateControls();
  353. throw;
  354. }
  355. }
  356. }
  357. //---------------------------------------------------------------------------
  358. void __fastcall TSynchronizeDialog::StopButtonClick(TObject * /*Sender*/)
  359. {
  360. Stop();
  361. }
  362. //---------------------------------------------------------------------------
  363. void __fastcall TSynchronizeDialog::OnlyStop()
  364. {
  365. FSynchronizing = false;
  366. DoStartStop(false, false);
  367. }
  368. //---------------------------------------------------------------------------
  369. void __fastcall TSynchronizeDialog::Stop()
  370. {
  371. OnlyStop();
  372. UpdateControls();
  373. Repaint();
  374. if (IsApplicationMinimized() && FMinimizedByMe)
  375. {
  376. FMinimizedByMe = false;
  377. Application->Restore();
  378. Application->BringToFront();
  379. }
  380. }
  381. //---------------------------------------------------------------------------
  382. void __fastcall TSynchronizeDialog::MinimizeButtonClick(TObject * Sender)
  383. {
  384. CallGlobalMinimizeHandler(Sender);
  385. }
  386. //---------------------------------------------------------------------------
  387. void __fastcall TSynchronizeDialog::GlobalMinimize(TObject * /*Sender*/)
  388. {
  389. Application->Minimize();
  390. FMinimizedByMe = true;
  391. }
  392. //---------------------------------------------------------------------------
  393. void __fastcall TSynchronizeDialog::SetSaveSettings(bool value)
  394. {
  395. SaveSettingsCheck->Checked = value;
  396. }
  397. //---------------------------------------------------------------------------
  398. bool __fastcall TSynchronizeDialog::GetSaveSettings()
  399. {
  400. return SaveSettingsCheck->Checked;
  401. }
  402. //---------------------------------------------------------------------------
  403. void __fastcall TSynchronizeDialog::FormShow(TObject * /*Sender*/)
  404. {
  405. InstallPathWordBreakProc(LocalDirectoryEdit);
  406. InstallPathWordBreakProc(RemoteDirectoryEdit);
  407. // OnShow gets called more than once sometimes
  408. if (!FSynchronizing)
  409. {
  410. ClearLog();
  411. UpdateControls();
  412. if (FStartImmediatelly)
  413. {
  414. StartButtonClick(NULL);
  415. }
  416. }
  417. else
  418. {
  419. assert(FStartImmediatelly);
  420. }
  421. }
  422. //---------------------------------------------------------------------------
  423. void __fastcall TSynchronizeDialog::FormCloseQuery(TObject * /*Sender*/,
  424. bool & /*CanClose*/)
  425. {
  426. if (FSynchronizing)
  427. {
  428. Stop();
  429. }
  430. }
  431. //---------------------------------------------------------------------------
  432. TCopyParamType __fastcall TSynchronizeDialog::GetCopyParams()
  433. {
  434. TCopyParamType Result = FCopyParams;
  435. Result.PreserveTime = true;
  436. Result.NewerOnly = false;
  437. return Result;
  438. }
  439. //---------------------------------------------------------------------------
  440. void __fastcall TSynchronizeDialog::SetCopyParams(const TCopyParamType & value)
  441. {
  442. FCopyParams = value;
  443. UpdateControls();
  444. }
  445. //---------------------------------------------------------------------------
  446. int __fastcall TSynchronizeDialog::ActualCopyParamAttrs()
  447. {
  448. return FCopyParamAttrs | cpaNoPreserveTime | cpaNoNewerOnly;
  449. }
  450. //---------------------------------------------------------------------------
  451. void __fastcall TSynchronizeDialog::CopyParamClick(TObject * Sender)
  452. {
  453. assert(FLAGCLEAR(FOptions, soDoNotUsePresets));
  454. // PreserveTime is forced for some settings, but avoid hard-setting it until
  455. // user really confirms it on custom dialog
  456. TCopyParamType ACopyParams = CopyParams;
  457. if (CopyParamListPopupClick(Sender, ACopyParams, FPreset,
  458. ActualCopyParamAttrs()))
  459. {
  460. FCopyParams = ACopyParams;
  461. UpdateControls();
  462. }
  463. }
  464. //---------------------------------------------------------------------------
  465. void __fastcall TSynchronizeDialog::CopyParamGroupContextPopup(
  466. TObject * /*Sender*/, TPoint & MousePos, bool & Handled)
  467. {
  468. if (FLAGCLEAR(FOptions, soDoNotUsePresets))
  469. {
  470. CopyParamListPopup(CalculatePopupRect(CopyParamGroup, MousePos),
  471. cplCustomizeDefault);
  472. Handled = true;
  473. }
  474. }
  475. //---------------------------------------------------------------------------
  476. void __fastcall TSynchronizeDialog::CopyParamGroupClick(TObject * /*Sender*/)
  477. {
  478. // PreserveTime is forced for some settings, but avoid hard-setting it until
  479. // user really confirms it on cutom dialog
  480. TCopyParamType ACopyParams = CopyParams;
  481. if (DoCopyParamCustomDialog(ACopyParams, ActualCopyParamAttrs()))
  482. {
  483. FCopyParams = ACopyParams;
  484. UpdateControls();
  485. }
  486. }
  487. //---------------------------------------------------------------------------
  488. void __fastcall TSynchronizeDialog::HelpButtonClick(TObject * /*Sender*/)
  489. {
  490. FormHelp(this);
  491. }
  492. //---------------------------------------------------------------------------
  493. void __fastcall TSynchronizeDialog::ClearLog()
  494. {
  495. // TListItems::Clear() does nothing without allocated handle
  496. LogView->HandleNeeded();
  497. LogView->Items->Clear();
  498. }
  499. //---------------------------------------------------------------------------
  500. void __fastcall TSynchronizeDialog::CopyLog()
  501. {
  502. TInstantOperationVisualizer Visualizer;
  503. UnicodeString Content;
  504. for (int i = 0; i < LogView->Items->Count; i++)
  505. {
  506. TListItem * Item = LogView->Items->Item[i];
  507. Content += Item->Caption + L"\t" + Item->SubItems->Strings[0] + L"\r\n";
  508. }
  509. CopyToClipboard(Content);
  510. }
  511. //---------------------------------------------------------------------------
  512. void __fastcall TSynchronizeDialog::LogViewKeyDown(TObject * /*Sender*/,
  513. WORD & Key, TShiftState Shift)
  514. {
  515. if (Key == VK_DELETE)
  516. {
  517. ClearLog();
  518. Key = 0;
  519. }
  520. else if ((Key == L'C') && Shift.Contains(ssCtrl) && (LogView->Items->Count > 0))
  521. {
  522. CopyLog();
  523. Key = 0;
  524. }
  525. }
  526. //---------------------------------------------------------------------------
  527. void __fastcall TSynchronizeDialog::FormKeyDown(TObject * /*Sender*/, WORD & Key,
  528. TShiftState /*Shift*/)
  529. {
  530. if ((Key == VK_ESCAPE) && FSynchronizing)
  531. {
  532. Stop();
  533. Key = 0;
  534. }
  535. }
  536. //---------------------------------------------------------------------------
  537. void __fastcall TSynchronizeDialog::TransferSettingsButtonDropDownClick(TObject * /*Sender*/)
  538. {
  539. CopyParamListPopup(CalculatePopupRect(TransferSettingsButton), cplCustomizeDefault);
  540. }
  541. //---------------------------------------------------------------------------