mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 14:12:44 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			182 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			182 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2022, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/LexicalPath.h>
 | |
| #include <AK/NumberFormat.h>
 | |
| #include <LibGUI/BoxLayout.h>
 | |
| #include <LibGUI/Button.h>
 | |
| #include <LibGUI/ImageWidget.h>
 | |
| #include <LibGUI/Label.h>
 | |
| #include <LibGUI/MessageBox.h>
 | |
| #include <LibGfx/Font/Font.h>
 | |
| 
 | |
| namespace GUI {
 | |
| 
 | |
| Dialog::ExecResult MessageBox::show(Window* parent_window, StringView text, StringView title, Type type, InputType input_type)
 | |
| {
 | |
|     auto box = MessageBox::construct(parent_window, text, title, type, input_type);
 | |
|     if (parent_window)
 | |
|         box->set_icon(parent_window->icon());
 | |
|     return box->exec();
 | |
| }
 | |
| 
 | |
| Dialog::ExecResult MessageBox::show_error(Window* parent_window, StringView text)
 | |
| {
 | |
|     return show(parent_window, text, "Error"sv, GUI::MessageBox::Type::Error, GUI::MessageBox::InputType::OK);
 | |
| }
 | |
| 
 | |
| Dialog::ExecResult MessageBox::ask_about_unsaved_changes(Window* parent_window, StringView path, Optional<Time> last_unmodified_timestamp)
 | |
| {
 | |
|     StringBuilder builder;
 | |
|     builder.append("Save changes to "sv);
 | |
|     if (path.is_empty())
 | |
|         builder.append("untitled document"sv);
 | |
|     else
 | |
|         builder.appendff("\"{}\"", LexicalPath::basename(path));
 | |
|     builder.append(" before closing?"sv);
 | |
| 
 | |
|     if (!path.is_empty() && last_unmodified_timestamp.has_value()) {
 | |
|         auto age = (Time::now_monotonic() - *last_unmodified_timestamp).to_seconds();
 | |
|         auto readable_time = human_readable_time(age);
 | |
|         builder.appendff("\nLast saved {} ago.", readable_time);
 | |
|     }
 | |
| 
 | |
|     auto box = MessageBox::construct(parent_window, builder.string_view(), "Unsaved changes"sv, Type::Warning, InputType::YesNoCancel);
 | |
|     if (parent_window)
 | |
|         box->set_icon(parent_window->icon());
 | |
| 
 | |
|     if (path.is_empty())
 | |
|         box->m_yes_button->set_text("Save As..."_string.release_value_but_fixme_should_propagate_errors());
 | |
|     else
 | |
|         box->m_yes_button->set_text("Save"_short_string);
 | |
|     box->m_no_button->set_text("Discard"_short_string);
 | |
|     box->m_cancel_button->set_text("Cancel"_short_string);
 | |
| 
 | |
|     return box->exec();
 | |
| }
 | |
| 
 | |
| void MessageBox::set_text(DeprecatedString text)
 | |
| {
 | |
|     m_text = move(text);
 | |
|     build();
 | |
| }
 | |
| 
 | |
| MessageBox::MessageBox(Window* parent_window, StringView text, StringView title, Type type, InputType input_type)
 | |
|     : Dialog(parent_window)
 | |
|     , m_text(text)
 | |
|     , m_type(type)
 | |
|     , m_input_type(input_type)
 | |
| {
 | |
|     set_title(title);
 | |
|     build();
 | |
| }
 | |
| 
 | |
| RefPtr<Gfx::Bitmap> MessageBox::icon() const
 | |
| {
 | |
|     switch (m_type) {
 | |
|     case Type::Information:
 | |
|         return Gfx::Bitmap::load_from_file("/res/icons/32x32/msgbox-information.png"sv).release_value_but_fixme_should_propagate_errors();
 | |
|     case Type::Warning:
 | |
|         return Gfx::Bitmap::load_from_file("/res/icons/32x32/msgbox-warning.png"sv).release_value_but_fixme_should_propagate_errors();
 | |
|     case Type::Error:
 | |
|         return Gfx::Bitmap::load_from_file("/res/icons/32x32/msgbox-error.png"sv).release_value_but_fixme_should_propagate_errors();
 | |
|     case Type::Question:
 | |
|         return Gfx::Bitmap::load_from_file("/res/icons/32x32/msgbox-question.png"sv).release_value_but_fixme_should_propagate_errors();
 | |
|     default:
 | |
|         return nullptr;
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool MessageBox::should_include_ok_button() const
 | |
| {
 | |
|     return m_input_type == InputType::OK || m_input_type == InputType::OKCancel;
 | |
| }
 | |
| 
 | |
| bool MessageBox::should_include_cancel_button() const
 | |
| {
 | |
|     return m_input_type == InputType::OKCancel || m_input_type == InputType::YesNoCancel;
 | |
| }
 | |
| 
 | |
| bool MessageBox::should_include_yes_button() const
 | |
| {
 | |
|     return m_input_type == InputType::YesNo || m_input_type == InputType::YesNoCancel;
 | |
| }
 | |
| 
 | |
| bool MessageBox::should_include_no_button() const
 | |
| {
 | |
|     return should_include_yes_button();
 | |
| }
 | |
| 
 | |
| void MessageBox::build()
 | |
| {
 | |
|     auto widget = set_main_widget<Widget>().release_value_but_fixme_should_propagate_errors();
 | |
| 
 | |
|     int text_width = widget->font().width(m_text);
 | |
|     auto number_of_lines = m_text.split('\n').size();
 | |
|     int padded_text_height = widget->font().pixel_size_rounded_up() * 1.6;
 | |
|     int total_text_height = number_of_lines * padded_text_height;
 | |
|     int icon_width = 0;
 | |
| 
 | |
|     widget->set_layout<VerticalBoxLayout>(8, 6);
 | |
|     widget->set_fill_with_background_color(true);
 | |
| 
 | |
|     auto& message_container = widget->add<Widget>();
 | |
|     message_container.set_layout<HorizontalBoxLayout>(GUI::Margins {}, 8);
 | |
| 
 | |
|     if (m_type != Type::None) {
 | |
|         auto& icon_image = message_container.add<ImageWidget>();
 | |
|         icon_image.set_bitmap(icon());
 | |
|         if (icon()) {
 | |
|             icon_width = icon()->width();
 | |
|             if (icon_width > 0)
 | |
|                 message_container.layout()->set_margins({ 0, 0, 0, 8 });
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     auto& label = message_container.add<Label>(m_text);
 | |
|     label.set_fixed_height(total_text_height);
 | |
|     if (m_type != Type::None)
 | |
|         label.set_text_alignment(Gfx::TextAlignment::CenterLeft);
 | |
| 
 | |
|     auto& button_container = widget->add<Widget>();
 | |
|     button_container.set_layout<HorizontalBoxLayout>(GUI::Margins {}, 8);
 | |
|     button_container.set_fixed_height(24);
 | |
| 
 | |
|     constexpr int button_width = 80;
 | |
|     int button_count = 0;
 | |
| 
 | |
|     auto add_button = [&](String label, ExecResult result) -> GUI::Button& {
 | |
|         auto& button = button_container.add<Button>();
 | |
|         button.set_fixed_width(button_width);
 | |
|         button.set_text(move(label));
 | |
|         button.on_click = [this, result](auto) {
 | |
|             done(result);
 | |
|         };
 | |
|         ++button_count;
 | |
|         return button;
 | |
|     };
 | |
| 
 | |
|     button_container.add_spacer().release_value_but_fixme_should_propagate_errors();
 | |
|     if (should_include_ok_button())
 | |
|         m_ok_button = add_button("OK"_short_string, ExecResult::OK);
 | |
|     if (should_include_yes_button())
 | |
|         m_yes_button = add_button("Yes"_short_string, ExecResult::Yes);
 | |
|     if (should_include_no_button())
 | |
|         m_no_button = add_button("No"_short_string, ExecResult::No);
 | |
|     if (should_include_cancel_button())
 | |
|         m_cancel_button = add_button("Cancel"_short_string, ExecResult::Cancel);
 | |
|     button_container.add_spacer().release_value_but_fixme_should_propagate_errors();
 | |
| 
 | |
|     int width = (button_count * button_width) + ((button_count - 1) * button_container.layout()->spacing()) + 32;
 | |
|     width = max(width, text_width + icon_width + 56);
 | |
| 
 | |
|     // FIXME: Use shrink from new layout system
 | |
|     set_rect(x(), y(), width, 80 + label.text_calculated_preferred_height());
 | |
|     set_resizable(false);
 | |
| }
 | |
| 
 | |
| }
 | 
