Synchronize.cpp 18 KB

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