mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 07:52:45 +00:00 
			
		
		
		
	 6e19ab2bbc
			
		
	
	
		6e19ab2bbc
		
	
	
	
	
		
			
			We have a new, improved string type coming up in AK (OOM aware, no null state), and while it's going to use UTF-8, the name UTF8String is a mouthful - so let's free up the String name by renaming the existing class. Making the old one have an annoying name will hopefully also help with quick adoption :^)
		
			
				
	
	
		
			190 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			190 lines
		
	
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
 | |
|  * Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
 | |
|  * Copyright (c) 2022, the SerenityOS developers.
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <LibGUI/BoxLayout.h>
 | |
| #include <LibGUI/Breadcrumbbar.h>
 | |
| #include <LibGUI/Button.h>
 | |
| #include <LibGUI/Painter.h>
 | |
| #include <LibGfx/Font/Font.h>
 | |
| #include <LibGfx/Palette.h>
 | |
| 
 | |
| REGISTER_WIDGET(GUI, Breadcrumbbar)
 | |
| 
 | |
| namespace GUI {
 | |
| 
 | |
| class BreadcrumbButton : public Button {
 | |
|     C_OBJECT(BreadcrumbButton);
 | |
| 
 | |
| public:
 | |
|     virtual ~BreadcrumbButton() override = default;
 | |
| 
 | |
|     virtual bool is_uncheckable() const override { return false; }
 | |
|     virtual void drop_event(DropEvent& event) override
 | |
|     {
 | |
|         if (on_drop)
 | |
|             on_drop(event);
 | |
|     }
 | |
| 
 | |
|     virtual void drag_enter_event(DragEvent& event) override
 | |
|     {
 | |
|         update();
 | |
|         if (on_drag_enter)
 | |
|             on_drag_enter(event);
 | |
|     }
 | |
| 
 | |
|     virtual void drag_leave_event(Event&) override
 | |
|     {
 | |
|         update();
 | |
|     }
 | |
| 
 | |
|     virtual void paint_event(PaintEvent& event) override
 | |
|     {
 | |
|         Button::paint_event(event);
 | |
|         if (has_pending_drop()) {
 | |
|             Painter painter(*this);
 | |
|             painter.draw_rect(rect(), palette().selection(), true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Function<void(DropEvent&)> on_drop;
 | |
|     Function<void(DragEvent&)> on_drag_enter;
 | |
| 
 | |
| private:
 | |
|     BreadcrumbButton() = default;
 | |
| };
 | |
| 
 | |
| Breadcrumbbar::Breadcrumbbar()
 | |
| {
 | |
|     auto& layout = set_layout<HorizontalBoxLayout>();
 | |
|     layout.set_spacing(0);
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::clear_segments()
 | |
| {
 | |
|     m_segments.clear();
 | |
|     remove_all_children();
 | |
|     m_selected_segment = {};
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::append_segment(DeprecatedString text, Gfx::Bitmap const* icon, DeprecatedString data, DeprecatedString tooltip)
 | |
| {
 | |
|     auto& button = add<BreadcrumbButton>();
 | |
|     button.set_button_style(Gfx::ButtonStyle::Coolbar);
 | |
|     button.set_text(text);
 | |
|     button.set_icon(icon);
 | |
|     button.set_tooltip(move(tooltip));
 | |
|     button.set_focus_policy(FocusPolicy::TabFocus);
 | |
|     button.set_checkable(true);
 | |
|     button.set_exclusive(true);
 | |
|     button.on_click = [this, index = m_segments.size()](auto) {
 | |
|         if (on_segment_click)
 | |
|             on_segment_click(index);
 | |
|         if (on_segment_change && m_selected_segment != index)
 | |
|             on_segment_change(index);
 | |
|     };
 | |
|     button.on_focus_change = [this, index = m_segments.size()](auto has_focus, auto) {
 | |
|         if (has_focus && on_segment_change && m_selected_segment != index)
 | |
|             on_segment_change(index);
 | |
|     };
 | |
|     button.on_drop = [this, index = m_segments.size()](auto& drop_event) {
 | |
|         if (on_segment_drop)
 | |
|             on_segment_drop(index, drop_event);
 | |
|     };
 | |
|     button.on_drag_enter = [this, index = m_segments.size()](auto& event) {
 | |
|         if (on_segment_drag_enter)
 | |
|             on_segment_drag_enter(index, event);
 | |
|     };
 | |
| 
 | |
|     auto button_text_width = button.font().width(text);
 | |
|     auto icon_width = icon ? icon->width() : 0;
 | |
|     auto icon_padding = icon ? 4 : 0;
 | |
| 
 | |
|     int const max_button_width = 100;
 | |
| 
 | |
|     auto button_width = min(button_text_width + icon_width + icon_padding + 16, max_button_width);
 | |
|     auto shrunken_width = icon_width + icon_padding + (icon ? 4 : 16);
 | |
| 
 | |
|     button.set_max_size(button_width, 16 + 8);
 | |
|     button.set_min_size(shrunken_width, 16 + 8);
 | |
| 
 | |
|     Segment segment { icon, text, data, button_width, shrunken_width, button.make_weak_ptr<GUI::Button>() };
 | |
| 
 | |
|     m_segments.append(move(segment));
 | |
|     relayout();
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::remove_end_segments(size_t start_segment_index)
 | |
| {
 | |
|     while (segment_count() > start_segment_index) {
 | |
|         auto segment = m_segments.take_last();
 | |
|         remove_child(*segment.button);
 | |
|     }
 | |
|     if (m_selected_segment.has_value() && *m_selected_segment >= start_segment_index)
 | |
|         m_selected_segment = {};
 | |
| }
 | |
| 
 | |
| Optional<size_t> Breadcrumbbar::find_segment_with_data(DeprecatedString const& data)
 | |
| {
 | |
|     for (size_t i = 0; i < segment_count(); ++i) {
 | |
|         if (segment_data(i) == data)
 | |
|             return i;
 | |
|     }
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::set_selected_segment(Optional<size_t> index)
 | |
| {
 | |
|     if (m_selected_segment == index)
 | |
|         return;
 | |
|     m_selected_segment = index;
 | |
| 
 | |
|     if (!index.has_value()) {
 | |
|         for_each_child_of_type<GUI::AbstractButton>([&](auto& button) {
 | |
|             button.set_checked(false);
 | |
|             return IterationDecision::Continue;
 | |
|         });
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     auto& segment = m_segments[index.value()];
 | |
|     VERIFY(segment.button);
 | |
|     segment.button->set_checked(true);
 | |
|     if (on_segment_change)
 | |
|         on_segment_change(index);
 | |
|     relayout();
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::doubleclick_event(MouseEvent& event)
 | |
| {
 | |
|     if (on_doubleclick)
 | |
|         on_doubleclick(event);
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::resize_event(ResizeEvent&)
 | |
| {
 | |
|     relayout();
 | |
| }
 | |
| 
 | |
| void Breadcrumbbar::relayout()
 | |
| {
 | |
|     auto remaining_width = 0;
 | |
| 
 | |
|     for (auto& segment : m_segments)
 | |
|         remaining_width += segment.width;
 | |
| 
 | |
|     for (auto& segment : m_segments) {
 | |
|         if (remaining_width > width() && !segment.button->is_checked()) {
 | |
|             segment.button->set_preferred_width(segment.shrunken_width);
 | |
|             remaining_width -= (segment.width - segment.shrunken_width);
 | |
|             continue;
 | |
|         }
 | |
|         segment.button->set_preferred_width(segment.width);
 | |
|     }
 | |
| }
 | |
| 
 | |
| }
 |