FileFind.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571
  1. //---------------------------------------------------------------------------
  2. #include <vcl.h>
  3. #pragma hdrstop
  4. #include <Common.h>
  5. #include <WinInterface.h>
  6. #include <VCLCommon.h>
  7. #include <TextsWin.h>
  8. #include <WinConfiguration.h>
  9. #include <CoreMain.h>
  10. #include <Tools.h>
  11. #include <BaseUtils.hpp>
  12. #include <Terminal.h>
  13. #include "FileFind.h"
  14. //---------------------------------------------------------------------------
  15. #pragma package(smart_init)
  16. #pragma link "HistoryComboBox"
  17. #pragma link "IEListView"
  18. #pragma link "NortonLikeListView"
  19. #pragma link "PngImageList"
  20. #ifndef NO_RESOURCES
  21. #pragma resource "*.dfm"
  22. #endif
  23. //---------------------------------------------------------------------------
  24. TFileFindDialog * FileFindDialog = NULL;
  25. //---------------------------------------------------------------------------
  26. void __fastcall ShowFileFindDialog(
  27. TTerminal * Terminal, UnicodeString Directory, TFindEvent OnFind, TFocusFileEvent OnFocusFile, TDeleteFilesEvent OnDeleteFiles)
  28. {
  29. if (FileFindDialog == NULL)
  30. {
  31. FileFindDialog = new TFileFindDialog(Application);
  32. }
  33. FileFindDialog->Init(Terminal, Directory, OnFind, OnFocusFile, OnDeleteFiles);
  34. FileFindDialog->Show();
  35. }
  36. //---------------------------------------------------------------------------
  37. void __fastcall HideFileFindDialog()
  38. {
  39. if (FileFindDialog != NULL)
  40. {
  41. delete FileFindDialog;
  42. }
  43. }
  44. //---------------------------------------------------------------------------
  45. __fastcall TFileFindDialog::TFileFindDialog(TComponent * Owner)
  46. : TForm(Owner)
  47. {
  48. UseSystemSettings(this);
  49. FState = ffInit;
  50. FixComboBoxResizeBug(MaskEdit);
  51. FixComboBoxResizeBug(RemoteDirectoryEdit);
  52. HintLabel(MaskHintText,
  53. FORMAT(L"%s\n \n%s\n \n%s\n \n%s\n \n%s\n \n%s", (LoadStr(MASK_HINT2),
  54. LoadStr(FILE_MASK_EX_HINT), LoadStr(COMBINING_MASKS_HINT),
  55. LoadStr(PATH_MASK_HINT2), LoadStr(DIRECTORY_MASK_HINT),
  56. LoadStr(MASK_HELP))));
  57. FSystemImageList = SharedSystemImageList(false);
  58. FileView->SmallImages = FSystemImageList;
  59. FileView->ShowColumnIcon = false;
  60. UseDesktopFont(FileView);
  61. UseDesktopFont(StatusBar);
  62. FFrameAnimation.Init(AnimationPaintBox, L"Find");
  63. FixFormIcons(this);
  64. }
  65. //---------------------------------------------------------------------------
  66. __fastcall TFileFindDialog::~TFileFindDialog()
  67. {
  68. TFindFileConfiguration FormConfiguration = CustomWinConfiguration->FindFile;
  69. FormConfiguration.ListParams = FileView->ColProperties->ParamsStr;
  70. UnicodeString WindowParams = StoreFormSize(this);
  71. // this is particularly to prevent saving the form state
  72. // for the first time, keeping default positioning by a system
  73. if (!FWindowParams.IsEmpty() && (FWindowParams != WindowParams))
  74. {
  75. FormConfiguration.WindowParams = WindowParams;
  76. }
  77. CustomWinConfiguration->FindFile = FormConfiguration;
  78. Clear();
  79. delete FSystemImageList;
  80. DebugAssert(FileFindDialog == this);
  81. FileFindDialog = NULL;
  82. }
  83. //---------------------------------------------------------------------------
  84. bool __fastcall TFileFindDialog::IsFinding()
  85. {
  86. return (FState == ffFinding) || (FState == ffAborting);
  87. }
  88. //---------------------------------------------------------------------------
  89. void __fastcall TFileFindDialog::UpdateControls()
  90. {
  91. bool Finding = IsFinding();
  92. Caption = FORMAT("%s - %s", (LoadStr(Finding ? FIND_FILE_FINDING : FIND_FILE_TITLE), FTerminalName));
  93. UnicodeString StartStopCaption;
  94. if (Finding)
  95. {
  96. EnableControl(StartStopButton, true);
  97. StartStopCaption = LoadStr(FIND_FILE_STOP);
  98. }
  99. else
  100. {
  101. EnableControl(StartStopButton, !RemoteDirectoryEdit->Text.IsEmpty());
  102. StartStopCaption = LoadStr(FIND_FILE_START);
  103. }
  104. StartStopButton->Caption = StartStopCaption;
  105. EnableControl(FilterGroup, !Finding);
  106. FocusAction->Enabled = (FileView->ItemFocused != NULL);
  107. bool EnableFileOperations = !Finding && (FileView->SelCount > 0);
  108. DeleteAction->Enabled = EnableFileOperations;
  109. CopyAction->Enabled = (FileView->Items->Count > 0);
  110. SelectAllAction->Enabled = (FileView->SelCount < FileView->Items->Count);
  111. switch (FState)
  112. {
  113. case ffInit:
  114. StatusBar->SimpleText = L"";
  115. break;
  116. case ffFinding:
  117. case ffAborting:
  118. if (!FFindingInDirectory.IsEmpty())
  119. {
  120. StatusBar->SimpleText = FMTLOAD(FIND_FILE_IN_DIRECTORY, (FFindingInDirectory));
  121. }
  122. else
  123. {
  124. StatusBar->SimpleText = L"";
  125. }
  126. break;
  127. case ffAborted:
  128. StatusBar->SimpleText = LoadStr(FIND_FILE_ABORTED);
  129. break;
  130. case ffDone:
  131. StatusBar->SimpleText = LoadStr(FIND_FILE_DONE);
  132. break;
  133. default:
  134. DebugFail();
  135. break;
  136. }
  137. FocusButton->Default = FileView->Focused() && (FState != ffInit);
  138. StartStopButton->Default = !FocusButton->Default;
  139. }
  140. //---------------------------------------------------------------------------
  141. void __fastcall TFileFindDialog::ControlChange(TObject * /*Sender*/)
  142. {
  143. UpdateControls();
  144. }
  145. //---------------------------------------------------------------------------
  146. void __fastcall TFileFindDialog::Init(
  147. TTerminal * Terminal, UnicodeString Directory, TFindEvent OnFind, TFocusFileEvent OnFocusFile, TDeleteFilesEvent OnDeleteFiles)
  148. {
  149. if (FTerminal != Terminal)
  150. {
  151. FTerminal = Terminal;
  152. FTerminalName = Terminal->SessionData->SessionName;
  153. Clear();
  154. FState = ffInit;
  155. ActiveControl = MaskEdit;
  156. }
  157. FOnFind = OnFind;
  158. FOnFocusFile = OnFocusFile;
  159. FOnDeleteFiles = OnDeleteFiles;
  160. MaskEdit->Text = WinConfiguration->SelectMask;
  161. RemoteDirectoryEdit->Text = UnixExcludeTrailingBackslash(Directory);
  162. UpdateControls();
  163. }
  164. //---------------------------------------------------------------------------
  165. void __fastcall TFileFindDialog::CreateParams(TCreateParams & Params)
  166. {
  167. TForm::CreateParams(Params);
  168. Params.WndParent = GetDesktopWindow();
  169. }
  170. //---------------------------------------------------------------------------
  171. void __fastcall TFileFindDialog::ClearItem(TListItem * Item)
  172. {
  173. TRemoteFile * File = static_cast<TRemoteFile *>(Item->Data);
  174. Item->Data = NULL;
  175. delete File;
  176. }
  177. //---------------------------------------------------------------------------
  178. void __fastcall TFileFindDialog::Clear()
  179. {
  180. for (int Index = 0; Index < FileView->Items->Count; Index++)
  181. {
  182. ClearItem(FileView->Items->Item[Index]);
  183. }
  184. FileView->Items->Clear();
  185. }
  186. //---------------------------------------------------------------------------
  187. void __fastcall TFileFindDialog::Start()
  188. {
  189. if (MaskEdit->Focused())
  190. {
  191. MaskEditExit(NULL);
  192. }
  193. RemoteDirectoryEdit->SaveToHistory();
  194. CustomWinConfiguration->History[L"RemoteDirectory"] = RemoteDirectoryEdit->Items;
  195. MaskEdit->SaveToHistory();
  196. WinConfiguration->History[L"Mask"] = MaskEdit->Items;
  197. WinConfiguration->SelectMask = MaskEdit->Text;
  198. DebugAssert(FState != ffFinding);
  199. FState = ffFinding;
  200. try
  201. {
  202. FFrameAnimation.Start();
  203. UpdateControls();
  204. Repaint();
  205. TOperationVisualizer Visualizer;
  206. DebugAssert(FOnFind != NULL);
  207. UnicodeString Directory = UnixExcludeTrailingBackslash(RemoteDirectoryEdit->Text);
  208. FDirectory = Directory;
  209. if (FDirectory == ROOTDIRECTORY)
  210. {
  211. FDirectory = UnicodeString();
  212. }
  213. FOnFind(FTerminal, Directory, MaskEdit->Text, FileFound, FindingFile);
  214. }
  215. __finally
  216. {
  217. FFindingInDirectory = L"";
  218. if (FState == ffFinding)
  219. {
  220. FState = ffDone;
  221. }
  222. if (FState == ffAborting)
  223. {
  224. FState = ffAborted;
  225. }
  226. FFrameAnimation.Stop();
  227. if (WindowState == wsMinimized)
  228. {
  229. ShowNotification(
  230. NULL, MainInstructions(LoadStr(BALLOON_OPERATION_COMPLETE)),
  231. qtInformation);
  232. }
  233. if (!FFocusPath.IsEmpty())
  234. {
  235. UnicodeString FocusPath = FFocusPath;
  236. FFocusPath = L"";
  237. DoFocusFile(FocusPath);
  238. }
  239. UpdateControls();
  240. }
  241. }
  242. //---------------------------------------------------------------------------
  243. void __fastcall TFileFindDialog::FileFound(TTerminal * /*Terminal*/,
  244. const UnicodeString FileName, const TRemoteFile * AFile, bool & Cancel)
  245. {
  246. TListItem * Item = FileView->Items->Add();
  247. TRemoteFile * File = AFile->Duplicate(true);
  248. Item->Data = File;
  249. Item->ImageIndex = File->IconIndex;
  250. UnicodeString Caption = File->FileName;
  251. if (File->IsDirectory)
  252. {
  253. Caption = UnixIncludeTrailingBackslash(Caption);
  254. }
  255. Item->Caption = Caption;
  256. UnicodeString Directory = UnixExtractFilePath(File->FullFileName);
  257. if (AnsiSameText(FDirectory, Directory.SubString(1, FDirectory.Length())))
  258. {
  259. Directory = L"." + Directory.SubString(FDirectory.Length() + 1, Directory.Length() - FDirectory.Length());
  260. }
  261. else
  262. {
  263. DebugFail();
  264. }
  265. Item->SubItems->Add(Directory);
  266. if (File->IsDirectory)
  267. {
  268. Item->SubItems->Add(L"");
  269. }
  270. else
  271. {
  272. Item->SubItems->Add(
  273. FormatPanelBytes(File->Size, WinConfiguration->FormatSizeBytes));
  274. }
  275. Item->SubItems->Add(UserModificationStr(File->Modification, File->ModificationFmt));
  276. UpdateControls();
  277. Cancel = (FState == ffAborting);
  278. Application->ProcessMessages();
  279. }
  280. //---------------------------------------------------------------------------
  281. void __fastcall TFileFindDialog::FindingFile(TTerminal * /*Terminal*/,
  282. const UnicodeString Directory, bool & Cancel)
  283. {
  284. if (!Directory.IsEmpty() && (FFindingInDirectory != Directory))
  285. {
  286. FFindingInDirectory = Directory;
  287. UpdateControls();
  288. }
  289. Cancel = (FState == ffAborting);
  290. Application->ProcessMessages();
  291. }
  292. //---------------------------------------------------------------------------
  293. void __fastcall TFileFindDialog::StartStopButtonClick(TObject * /*Sender*/)
  294. {
  295. if (IsFinding())
  296. {
  297. Stop();
  298. }
  299. else
  300. {
  301. Clear();
  302. if (ActiveControl->Parent == FilterGroup)
  303. {
  304. FileView->SetFocus();
  305. }
  306. Start();
  307. }
  308. }
  309. //---------------------------------------------------------------------------
  310. void __fastcall TFileFindDialog::StopButtonClick(TObject * /*Sender*/)
  311. {
  312. Stop();
  313. }
  314. //---------------------------------------------------------------------------
  315. void __fastcall TFileFindDialog::Stop()
  316. {
  317. FState = ffAborting;
  318. UpdateControls();
  319. }
  320. //---------------------------------------------------------------------------
  321. void __fastcall TFileFindDialog::FormShow(TObject * /*Sender*/)
  322. {
  323. InstallPathWordBreakProc(MaskEdit);
  324. InstallPathWordBreakProc(RemoteDirectoryEdit);
  325. // have to set history after value, to prevent autocompletition
  326. MaskEdit->Items = WinConfiguration->History[L"Mask"];
  327. RemoteDirectoryEdit->Items = CustomWinConfiguration->History[L"RemoteDirectory"];
  328. UpdateFormPosition(this, poOwnerFormCenter);
  329. RestoreFormSize(CustomWinConfiguration->FindFile.WindowParams, this);
  330. FileView->ColProperties->ParamsStr = CustomWinConfiguration->FindFile.ListParams;
  331. DebugAssert(FWindowParams.IsEmpty());
  332. if (FWindowParams.IsEmpty())
  333. {
  334. FWindowParams = StoreFormSize(this);
  335. }
  336. UpdateControls();
  337. }
  338. //---------------------------------------------------------------------------
  339. bool __fastcall TFileFindDialog::StopIfFinding()
  340. {
  341. bool Result = IsFinding();
  342. if (Result)
  343. {
  344. Stop();
  345. }
  346. return Result;
  347. }
  348. //---------------------------------------------------------------------------
  349. void __fastcall TFileFindDialog::FormCloseQuery(TObject * /*Sender*/,
  350. bool & /*CanClose*/)
  351. {
  352. StopIfFinding();
  353. }
  354. //---------------------------------------------------------------------------
  355. void __fastcall TFileFindDialog::HelpButtonClick(TObject * /*Sender*/)
  356. {
  357. FormHelp(this);
  358. }
  359. //---------------------------------------------------------------------------
  360. void __fastcall TFileFindDialog::Dispatch(void * Message)
  361. {
  362. TMessage * M = reinterpret_cast<TMessage*>(Message);
  363. if (M->Msg == CM_DIALOGKEY)
  364. {
  365. CMDialogKey(*((TWMKeyDown *)Message));
  366. }
  367. else
  368. {
  369. TForm::Dispatch(Message);
  370. }
  371. }
  372. //---------------------------------------------------------------------------
  373. void __fastcall TFileFindDialog::CMDialogKey(TWMKeyDown & Message)
  374. {
  375. // Handling VK_ESCAPE in FormKeyDown causes a beep when any "edit" has focus.
  376. // Moreover FormKeyDown is called when the the "esc" is pressed while drop down list is unrolled.
  377. if (Message.CharCode == VK_ESCAPE)
  378. {
  379. Close();
  380. Message.Result = 1;
  381. return;
  382. }
  383. TForm::Dispatch(&Message);
  384. }
  385. //---------------------------------------------------------------------------
  386. void __fastcall TFileFindDialog::FormKeyDown(TObject * /*Sender*/, WORD & Key,
  387. TShiftState Shift)
  388. {
  389. if ((Key == L'C') && Shift.Contains(ssCtrl) &&
  390. (dynamic_cast<TCustomCombo *>(ActiveControl) == NULL))
  391. {
  392. CopyToClipboard();
  393. Key = 0;
  394. }
  395. else if ((Key == VK_F10) && Shift.Empty())
  396. {
  397. Key = 0;
  398. StopIfFinding();
  399. Close();
  400. }
  401. }
  402. //---------------------------------------------------------------------------
  403. void __fastcall TFileFindDialog::MaskEditExit(TObject * /*Sender*/)
  404. {
  405. ValidateMaskEdit(MaskEdit);
  406. }
  407. //---------------------------------------------------------------------------
  408. void __fastcall TFileFindDialog::DoFocusFile(const UnicodeString & Path)
  409. {
  410. FOnFocusFile(FTerminal, Path);
  411. }
  412. //---------------------------------------------------------------------------
  413. void __fastcall TFileFindDialog::FocusFile()
  414. {
  415. UnicodeString Path = static_cast<TRemoteFile *>(FileView->ItemFocused->Data)->FullFileName;
  416. // To make focussing directories work,
  417. // otherwise it would try to focus "empty-named file" in the Path
  418. Path = UnixExcludeTrailingBackslash(Path);
  419. if (StopIfFinding())
  420. {
  421. FFocusPath = Path;
  422. }
  423. else
  424. {
  425. DoFocusFile(Path);
  426. }
  427. }
  428. //---------------------------------------------------------------------------
  429. void __fastcall TFileFindDialog::FileViewDblClick(TObject * /*Sender*/)
  430. {
  431. if (FileView->ItemFocused != NULL)
  432. {
  433. FocusFile();
  434. }
  435. }
  436. //---------------------------------------------------------------------------
  437. void __fastcall TFileFindDialog::FocusActionExecute(TObject * /*Sender*/)
  438. {
  439. FocusFile();
  440. }
  441. //---------------------------------------------------------------------------
  442. void __fastcall TFileFindDialog::FileViewSelectItem(TObject * /*Sender*/,
  443. TListItem * /*Item*/, bool /*Selected*/)
  444. {
  445. UpdateControls();
  446. }
  447. //---------------------------------------------------------------------------
  448. void __fastcall TFileFindDialog::MaskButtonClick(TObject * /*Sender*/)
  449. {
  450. TFileMasks Masks = MaskEdit->Text;
  451. if (DoEditMaskDialog(Masks))
  452. {
  453. MaskEdit->Text = Masks.Masks;
  454. }
  455. }
  456. //---------------------------------------------------------------------------
  457. void __fastcall TFileFindDialog::CopyToClipboard()
  458. {
  459. TInstantOperationVisualizer Visualizer;
  460. std::unique_ptr<TStrings> Strings(new TStringList());
  461. for (int Index = 0; Index < FileView->Items->Count; Index++)
  462. {
  463. TListItem * Item = FileView->Items->Item[Index];
  464. TRemoteFile * File = static_cast<TRemoteFile *>(Item->Data);
  465. Strings->Add(File->FullFileName);
  466. }
  467. ::CopyToClipboard(Strings.get());
  468. }
  469. //---------------------------------------------------------------------------
  470. void __fastcall TFileFindDialog::CopyActionExecute(TObject * /*Sender*/)
  471. {
  472. CopyToClipboard();
  473. }
  474. //---------------------------------------------------------------------------
  475. void __fastcall TFileFindDialog::FormClose(TObject * /*Sender*/, TCloseAction & Action)
  476. {
  477. StopIfFinding();
  478. Action = caFree;
  479. }
  480. //---------------------------------------------------------------------------
  481. void __fastcall TFileFindDialog::FileViewContextPopup(TObject * Sender, TPoint & MousePos, bool & Handled)
  482. {
  483. // to update source popup menu before TBX menu is created
  484. UpdateControls();
  485. MenuPopup(Sender, MousePos, Handled);
  486. }
  487. //---------------------------------------------------------------------------
  488. void __fastcall TFileFindDialog::FileDeleteFinished(const UnicodeString & FileName, bool Success)
  489. {
  490. TFileItemMap::iterator I = FFileItemMap.find(FileName);
  491. if (DebugAlwaysTrue(I != FFileItemMap.end()))
  492. {
  493. TListItem * Item = I->second;
  494. FileView->MakeProgressVisible(Item);
  495. if (Success)
  496. {
  497. ClearItem(Item);
  498. Item->Delete();
  499. }
  500. FFileItemMap.erase(I);
  501. }
  502. }
  503. //---------------------------------------------------------------------------
  504. void __fastcall TFileFindDialog::DeleteActionExecute(TObject * /*Sender*/)
  505. {
  506. std::unique_ptr<TStrings> FileList(new TStringList());
  507. DebugAssert(FFileItemMap.empty());
  508. TListItem * Item = FileView->Selected;
  509. while (Item != NULL)
  510. {
  511. TRemoteFile * File = static_cast<TRemoteFile *>(Item->Data);
  512. FileList->AddObject(File->FullFileName, File);
  513. FFileItemMap.insert(std::make_pair(File->FullFileName, Item));
  514. Item = FileView->GetNextItem(Item, sdAll, TItemStates() << isSelected);
  515. }
  516. try
  517. {
  518. FOnDeleteFiles(FTerminal, FileList.get(), FileDeleteFinished);
  519. }
  520. __finally
  521. {
  522. // can be non-empty only when not all files were deleted
  523. FFileItemMap.clear();
  524. }
  525. UpdateControls();
  526. }
  527. //---------------------------------------------------------------------------
  528. void __fastcall TFileFindDialog::SelectAllActionExecute(TObject * /*Sender*/)
  529. {
  530. FileView->SelectAll(smAll);
  531. }
  532. //---------------------------------------------------------------------------