/* * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include "FileOperationProgressWidget.h" #include #include #include #include #include #include #include #include #include namespace FileManager { FileOperationProgressWidget::FileOperationProgressWidget(NonnullRefPtr helper_pipe) : m_helper_pipe(move(helper_pipe)) { load_from_gml(file_operation_progress_gml); auto& button = *find_descendant_of_type_named("button"); auto& file_copy_animation = *find_descendant_of_type_named("file_copy_animation"); file_copy_animation.load_from_file("/res/graphics/file-flying-animation.gif"); file_copy_animation.animate(); auto& source_folder_icon = *find_descendant_of_type_named("source_folder_icon"); source_folder_icon.load_from_file("/res/icons/32x32/filetype-folder-open.png"); auto& destination_folder_icon = *find_descendant_of_type_named("destination_folder_icon"); destination_folder_icon.load_from_file("/res/icons/32x32/filetype-folder-open.png"); button.on_click = [this](auto) { close_pipe(); window()->close(); }; m_notifier = Core::Notifier::construct(m_helper_pipe->fd(), Core::Notifier::Read); m_notifier->on_ready_to_read = [this] { auto line = m_helper_pipe->read_line(); if (line.is_null()) { did_error("Read from pipe returned null."); return; } auto parts = line.split_view(' '); VERIFY(!parts.is_empty()); if (parts[0] == "ERROR"sv) { did_error(line.substring(6)); return; } if (parts[0] == "WARN"sv) { did_error(line.substring(5)); return; } if (parts[0] == "FINISH"sv) { did_finish(); return; } if (parts[0] == "PROGRESS"sv) { VERIFY(parts.size() >= 8); did_progress( parts[3].to_uint().value_or(0), parts[4].to_uint().value_or(0), parts[1].to_uint().value_or(0), parts[2].to_uint().value_or(0), parts[5].to_uint().value_or(0), parts[6].to_uint().value_or(0), parts[7]); } }; m_elapsed_timer.start(); } FileOperationProgressWidget::~FileOperationProgressWidget() { close_pipe(); } void FileOperationProgressWidget::did_finish() { close_pipe(); window()->close(); } void FileOperationProgressWidget::did_error(const String message) { // FIXME: Communicate more with the user about errors. close_pipe(); GUI::MessageBox::show(window(), String::formatted("An error occurred: {}", message), "Error", GUI::MessageBox::Type::Error, GUI::MessageBox::InputType::OK); window()->close(); } String FileOperationProgressWidget::estimate_time(off_t bytes_done, off_t total_byte_count) { int elapsed = m_elapsed_timer.elapsed() / 1000; if (bytes_done == 0 || elapsed < 3) return "Estimating..."; off_t bytes_left = total_byte_count - bytes_done; int seconds_remaining = (bytes_left * elapsed) / bytes_done; if (seconds_remaining < 30) return String::formatted("{} seconds", 5 + seconds_remaining - seconds_remaining % 5); if (seconds_remaining < 60) return "About a minute"; if (seconds_remaining < 90) return "Over a minute"; if (seconds_remaining < 120) return "Less than two minutes"; time_t minutes_remaining = seconds_remaining / 60; seconds_remaining %= 60; if (minutes_remaining < 60) { if (seconds_remaining < 30) return String::formatted("About {} minutes", minutes_remaining); return String::formatted("Over {} minutes", minutes_remaining); } time_t hours_remaining = minutes_remaining / 60; minutes_remaining %= 60; return String::formatted("{} hours and {} minutes", hours_remaining, minutes_remaining); } void FileOperationProgressWidget::did_progress(off_t bytes_done, off_t total_byte_count, size_t files_done, size_t total_file_count, [[maybe_unused]] off_t current_file_done, [[maybe_unused]] off_t current_file_size, const StringView& current_filename) { auto& files_copied_label = *find_descendant_of_type_named("files_copied_label"); auto& current_file_label = *find_descendant_of_type_named("current_file_label"); auto& overall_progressbar = *find_descendant_of_type_named("overall_progressbar"); auto& estimated_time_label = *find_descendant_of_type_named("estimated_time_label"); current_file_label.set_text(current_filename); files_copied_label.set_text(String::formatted("Copying file {} of {}", files_done, total_file_count)); estimated_time_label.set_text(estimate_time(bytes_done, total_byte_count)); if (total_byte_count) { window()->set_progress(100.0f * bytes_done / total_byte_count); overall_progressbar.set_max(total_byte_count); overall_progressbar.set_value(bytes_done); } } void FileOperationProgressWidget::close_pipe() { if (!m_helper_pipe) return; m_helper_pipe = nullptr; if (m_notifier) { m_notifier->set_enabled(false); m_notifier->on_ready_to_read = nullptr; } m_notifier = nullptr; } }