2021-04-12 21:32:40 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
|
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2021-04-12 21:32:40 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "FileOperationProgressWidget.h"
|
|
|
|
#include <Applications/FileManager/FileOperationProgressGML.h>
|
2021-04-13 10:58:09 +03:00
|
|
|
#include <LibCore/File.h>
|
2021-04-12 21:32:40 +03:00
|
|
|
#include <LibCore/Notifier.h>
|
|
|
|
#include <LibGUI/Button.h>
|
2021-04-17 04:39:35 +03:00
|
|
|
#include <LibGUI/ImageWidget.h>
|
2021-04-12 21:32:40 +03:00
|
|
|
#include <LibGUI/Label.h>
|
|
|
|
#include <LibGUI/MessageBox.h>
|
2021-04-13 17:18:20 +03:00
|
|
|
#include <LibGUI/Progressbar.h>
|
2021-04-12 21:32:40 +03:00
|
|
|
#include <LibGUI/Window.h>
|
|
|
|
|
|
|
|
namespace FileManager {
|
|
|
|
|
2021-04-13 10:58:09 +03:00
|
|
|
FileOperationProgressWidget::FileOperationProgressWidget(NonnullRefPtr<Core::File> helper_pipe)
|
|
|
|
: m_helper_pipe(move(helper_pipe))
|
2021-04-12 21:32:40 +03:00
|
|
|
{
|
|
|
|
load_from_gml(file_operation_progress_gml);
|
|
|
|
|
|
|
|
auto& button = *find_descendant_of_type_named<GUI::Button>("button");
|
|
|
|
|
2021-04-17 04:39:35 +03:00
|
|
|
auto& file_copy_animation = *find_descendant_of_type_named<GUI::ImageWidget>("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<GUI::ImageWidget>("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<GUI::ImageWidget>("destination_folder_icon");
|
|
|
|
destination_folder_icon.load_from_file("/res/icons/32x32/filetype-folder-open.png");
|
|
|
|
|
2021-04-12 21:32:40 +03:00
|
|
|
button.on_click = [this] {
|
|
|
|
close_pipe();
|
|
|
|
window()->close();
|
|
|
|
};
|
|
|
|
|
2021-04-13 10:58:09 +03:00
|
|
|
m_notifier = Core::Notifier::construct(m_helper_pipe->fd(), Core::Notifier::Read);
|
2021-04-12 21:32:40 +03:00
|
|
|
m_notifier->on_ready_to_read = [this] {
|
2021-04-13 10:58:09 +03:00
|
|
|
auto line = m_helper_pipe->read_line();
|
|
|
|
if (line.is_null()) {
|
2021-04-14 21:15:28 +03:00
|
|
|
did_error("Read from pipe returned null.");
|
2021-04-12 21:32:40 +03:00
|
|
|
return;
|
|
|
|
}
|
2021-04-14 21:15:28 +03:00
|
|
|
|
2021-04-13 10:58:09 +03:00
|
|
|
auto parts = line.split_view(' ');
|
2021-04-12 21:32:40 +03:00
|
|
|
VERIFY(!parts.is_empty());
|
|
|
|
|
2021-04-14 21:15:28 +03:00
|
|
|
if (parts[0] == "ERROR"sv) {
|
|
|
|
did_error(line.substring(6));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0] == "WARN"sv) {
|
|
|
|
did_error(line.substring(5));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-04-13 10:58:09 +03:00
|
|
|
if (parts[0] == "FINISH"sv) {
|
2021-04-12 21:32:40 +03:00
|
|
|
did_finish();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0] == "PROGRESS"sv) {
|
2021-04-13 11:08:59 +03:00
|
|
|
VERIFY(parts.size() >= 8);
|
2021-04-12 21:32:40 +03:00
|
|
|
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),
|
2021-04-13 11:08:59 +03:00
|
|
|
parts[5].to_uint().value_or(0),
|
|
|
|
parts[6].to_uint().value_or(0),
|
|
|
|
parts[7]);
|
2021-04-12 21:32:40 +03:00
|
|
|
}
|
|
|
|
};
|
2021-04-17 04:41:44 +03:00
|
|
|
|
|
|
|
m_elapsed_timer.start();
|
2021-04-12 21:32:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
FileOperationProgressWidget::~FileOperationProgressWidget()
|
|
|
|
{
|
|
|
|
close_pipe();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FileOperationProgressWidget::did_finish()
|
|
|
|
{
|
|
|
|
close_pipe();
|
|
|
|
window()->close();
|
|
|
|
}
|
|
|
|
|
2021-04-14 21:15:28 +03:00
|
|
|
void FileOperationProgressWidget::did_error(const String message)
|
2021-04-12 21:32:40 +03:00
|
|
|
{
|
|
|
|
// FIXME: Communicate more with the user about errors.
|
|
|
|
close_pipe();
|
2021-04-14 21:15:28 +03:00
|
|
|
GUI::MessageBox::show(window(), String::formatted("An error occurred: {}", message), "Error", GUI::MessageBox::Type::Error, GUI::MessageBox::InputType::OK);
|
2021-04-12 21:32:40 +03:00
|
|
|
window()->close();
|
|
|
|
}
|
|
|
|
|
2021-04-17 04:41:44 +03:00
|
|
|
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_file_name)
|
2021-04-12 21:32:40 +03:00
|
|
|
{
|
2021-04-17 04:41:44 +03:00
|
|
|
auto& files_copied_label = *find_descendant_of_type_named<GUI::Label>("files_copied_label");
|
2021-04-12 21:32:40 +03:00
|
|
|
auto& current_file_label = *find_descendant_of_type_named<GUI::Label>("current_file_label");
|
2021-04-13 17:18:20 +03:00
|
|
|
auto& overall_progressbar = *find_descendant_of_type_named<GUI::Progressbar>("overall_progressbar");
|
2021-04-17 04:41:44 +03:00
|
|
|
auto& estimated_time_label = *find_descendant_of_type_named<GUI::Label>("estimated_time_label");
|
2021-04-12 21:32:40 +03:00
|
|
|
|
|
|
|
current_file_label.set_text(current_file_name);
|
|
|
|
|
2021-04-17 04:41:44 +03:00
|
|
|
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);
|
|
|
|
}
|
2021-04-12 21:32:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileOperationProgressWidget::close_pipe()
|
|
|
|
{
|
|
|
|
if (!m_helper_pipe)
|
|
|
|
return;
|
|
|
|
m_helper_pipe = nullptr;
|
2021-04-14 21:05:23 +03:00
|
|
|
if (m_notifier) {
|
|
|
|
m_notifier->set_enabled(false);
|
2021-04-12 21:32:40 +03:00
|
|
|
m_notifier->on_ready_to_read = nullptr;
|
2021-04-14 21:05:23 +03:00
|
|
|
}
|
2021-04-12 21:32:40 +03:00
|
|
|
m_notifier = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|