Synchronize.cpp 17 KB

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