FileUtils.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "FileUtils.h"
  8. #include "FileOperationProgressWidget.h"
  9. #include <AK/LexicalPath.h>
  10. #include <LibCore/Stream.h>
  11. #include <LibCore/System.h>
  12. #include <LibGUI/MessageBox.h>
  13. #include <unistd.h>
  14. namespace FileManager {
  15. HashTable<NonnullRefPtr<GUI::Window>> file_operation_windows;
  16. void delete_paths(Vector<String> const& paths, bool should_confirm, GUI::Window* parent_window)
  17. {
  18. String message;
  19. if (paths.size() == 1) {
  20. message = String::formatted("Are you sure you want to delete {}?", LexicalPath::basename(paths[0]));
  21. } else {
  22. message = String::formatted("Are you sure you want to delete {} files?", paths.size());
  23. }
  24. if (should_confirm) {
  25. auto result = GUI::MessageBox::show(parent_window,
  26. message,
  27. "Confirm deletion",
  28. GUI::MessageBox::Type::Warning,
  29. GUI::MessageBox::InputType::OKCancel);
  30. if (result == GUI::MessageBox::ExecResult::Cancel)
  31. return;
  32. }
  33. if (run_file_operation(FileOperation::Delete, paths, {}, parent_window).is_error())
  34. _exit(1);
  35. }
  36. ErrorOr<void> run_file_operation(FileOperation operation, Vector<String> const& sources, String const& destination, GUI::Window* parent_window)
  37. {
  38. auto pipe_fds = TRY(Core::System::pipe2(0));
  39. pid_t child_pid = TRY(Core::System::fork());
  40. if (!child_pid) {
  41. TRY(Core::System::close(pipe_fds[0]));
  42. TRY(Core::System::dup2(pipe_fds[1], STDOUT_FILENO));
  43. Vector<StringView> file_operation_args;
  44. file_operation_args.append("/bin/FileOperation");
  45. switch (operation) {
  46. case FileOperation::Copy:
  47. file_operation_args.append("Copy");
  48. break;
  49. case FileOperation::Move:
  50. file_operation_args.append("Move");
  51. break;
  52. case FileOperation::Delete:
  53. file_operation_args.append("Delete");
  54. break;
  55. default:
  56. VERIFY_NOT_REACHED();
  57. }
  58. for (auto& source : sources)
  59. file_operation_args.append(source.view());
  60. if (operation != FileOperation::Delete)
  61. file_operation_args.append(destination.view());
  62. TRY(Core::System::exec(file_operation_args.first(), file_operation_args, Core::System::SearchInPath::Yes));
  63. VERIFY_NOT_REACHED();
  64. } else {
  65. TRY(Core::System::close(pipe_fds[1]));
  66. }
  67. auto window = TRY(GUI::Window::try_create());
  68. TRY(file_operation_windows.try_set(window));
  69. switch (operation) {
  70. case FileOperation::Copy:
  71. window->set_title("Copying Files...");
  72. break;
  73. case FileOperation::Move:
  74. window->set_title("Moving Files...");
  75. break;
  76. case FileOperation::Delete:
  77. window->set_title("Deleting Files...");
  78. break;
  79. default:
  80. VERIFY_NOT_REACHED();
  81. }
  82. auto pipe_input_file = TRY(Core::Stream::File::adopt_fd(pipe_fds[0], Core::Stream::OpenMode::Read));
  83. auto buffered_pipe = TRY(Core::Stream::BufferedFile::create(move(pipe_input_file)));
  84. (void)TRY(window->try_set_main_widget<FileOperationProgressWidget>(operation, move(buffered_pipe), pipe_fds[0]));
  85. window->resize(320, 190);
  86. if (parent_window)
  87. window->center_within(*parent_window);
  88. window->show();
  89. return {};
  90. }
  91. }