2020-01-18 09:38:21 +01:00
|
|
|
/*
|
2021-06-15 14:28:39 +02:00
|
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
2022-02-26 10:50:04 -07:00
|
|
|
* Copyright (c) 2022, the SerenityOS developers.
|
2020-01-18 09:38:21 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 09:38:21 +01:00
|
|
|
*/
|
|
|
|
|
|
2019-05-05 01:31:02 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
2021-06-15 14:28:39 +02:00
|
|
|
#include <LibGUI/Margins.h>
|
2020-02-06 20:33:02 +01:00
|
|
|
#include <LibGUI/Widget.h>
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
namespace GUI {
|
|
|
|
|
|
|
|
|
|
class TabWidget : public Widget {
|
|
|
|
|
C_OBJECT(TabWidget)
|
2019-05-05 01:31:02 +02:00
|
|
|
public:
|
2019-07-28 17:39:04 +02:00
|
|
|
enum TabPosition {
|
|
|
|
|
Top,
|
|
|
|
|
Bottom,
|
2022-05-17 20:47:35 +10:00
|
|
|
Left,
|
|
|
|
|
Right,
|
2019-07-28 17:39:04 +02:00
|
|
|
};
|
|
|
|
|
|
2022-02-26 10:50:04 -07:00
|
|
|
virtual ~TabWidget() override = default;
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2019-07-28 17:39:04 +02:00
|
|
|
TabPosition tab_position() const { return m_tab_position; }
|
|
|
|
|
void set_tab_position(TabPosition);
|
2022-05-17 20:47:35 +10:00
|
|
|
bool has_vertical_tabs() const { return m_tab_position == TabPosition::Left || m_tab_position == TabPosition::Right; }
|
2019-09-07 16:57:26 +02:00
|
|
|
|
2021-07-28 01:11:41 +01:00
|
|
|
Optional<size_t> active_tab_index() const;
|
2022-06-05 13:08:09 +02:00
|
|
|
size_t tab_count() { return m_tabs.size(); }
|
2019-07-28 17:39:04 +02:00
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
Widget* active_widget() { return m_active_widget.ptr(); }
|
2022-04-01 20:58:27 +03:00
|
|
|
Widget const* active_widget() const { return m_active_widget.ptr(); }
|
2020-02-02 15:07:41 +01:00
|
|
|
void set_active_widget(Widget*);
|
2021-01-08 01:01:23 +01:00
|
|
|
void set_tab_index(int);
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2022-05-17 20:47:35 +10:00
|
|
|
int bar_height() const { return m_bar_visible ? 22 : 0; }
|
|
|
|
|
|
|
|
|
|
int get_max_tab_width() const { return m_bar_visible ? m_max_tab_width : 0; }
|
|
|
|
|
void set_max_tab_width(int width) { m_max_tab_width = width; }
|
|
|
|
|
|
|
|
|
|
int get_min_tab_width() const { return m_min_tab_width; }
|
|
|
|
|
void set_min_tab_width(int width) { m_min_tab_width = width; }
|
2020-04-24 20:30:32 +02:00
|
|
|
|
2021-06-15 14:28:39 +02:00
|
|
|
GUI::Margins const& container_margins() const { return m_container_margins; }
|
|
|
|
|
void set_container_margins(GUI::Margins const&);
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2022-07-04 05:39:24 +02:00
|
|
|
Optional<UISize> calculated_min_size() const override;
|
|
|
|
|
Optional<UISize> calculated_preferred_size() const override;
|
|
|
|
|
|
2022-03-18 22:57:05 +01:00
|
|
|
ErrorOr<void> try_add_widget(Widget&);
|
2021-11-24 13:13:30 +01:00
|
|
|
|
2022-03-18 22:57:05 +01:00
|
|
|
void add_widget(Widget&);
|
2020-04-06 01:02:24 +03:00
|
|
|
void remove_widget(Widget&);
|
2020-02-23 12:23:48 +01:00
|
|
|
|
2021-11-24 13:13:30 +01:00
|
|
|
template<class T, class... Args>
|
2021-11-27 17:30:46 +01:00
|
|
|
ErrorOr<NonnullRefPtr<T>> try_add_tab(String title, Args&&... args)
|
2021-11-24 13:13:30 +01:00
|
|
|
{
|
|
|
|
|
auto t = TRY(T::try_create(forward<Args>(args)...));
|
2022-03-18 22:57:05 +01:00
|
|
|
t->set_title(move(title));
|
|
|
|
|
TRY(try_add_widget(*t));
|
2021-11-24 13:13:30 +01:00
|
|
|
return *t;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-23 12:23:48 +01:00
|
|
|
template<class T, class... Args>
|
2021-11-27 17:30:46 +01:00
|
|
|
T& add_tab(String title, Args&&... args)
|
2020-02-23 12:23:48 +01:00
|
|
|
{
|
|
|
|
|
auto t = T::construct(forward<Args>(args)...);
|
2022-03-18 22:57:05 +01:00
|
|
|
t->set_title(move(title));
|
|
|
|
|
add_widget(*t);
|
2020-04-04 11:10:07 +02:00
|
|
|
return *t;
|
2020-02-23 12:23:48 +01:00
|
|
|
}
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2020-04-06 01:02:24 +03:00
|
|
|
void remove_tab(Widget& tab) { remove_widget(tab); }
|
2021-08-06 10:01:32 +08:00
|
|
|
void remove_all_tabs_except(Widget& tab);
|
2020-04-06 01:02:24 +03:00
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
void set_tab_title(Widget& tab, StringView title);
|
2022-04-01 20:58:27 +03:00
|
|
|
void set_tab_icon(Widget& tab, Gfx::Bitmap const*);
|
2020-04-23 21:13:47 +02:00
|
|
|
|
2020-04-23 21:43:08 +02:00
|
|
|
void activate_next_tab();
|
|
|
|
|
void activate_previous_tab();
|
2022-03-16 20:06:27 +01:00
|
|
|
void activate_last_tab();
|
2020-04-23 21:43:08 +02:00
|
|
|
|
2020-04-24 22:27:46 +02:00
|
|
|
void set_text_alignment(Gfx::TextAlignment alignment) { m_text_alignment = alignment; }
|
|
|
|
|
Gfx::TextAlignment text_alignment() const { return m_text_alignment; }
|
|
|
|
|
|
2020-09-15 21:33:37 +02:00
|
|
|
bool uniform_tabs() const { return m_uniform_tabs; }
|
2020-04-24 22:36:25 +02:00
|
|
|
void set_uniform_tabs(bool uniform_tabs) { m_uniform_tabs = uniform_tabs; }
|
|
|
|
|
int uniform_tab_width() const;
|
|
|
|
|
|
2020-05-18 22:44:08 +01:00
|
|
|
void set_bar_visible(bool bar_visible);
|
|
|
|
|
bool is_bar_visible() const { return m_bar_visible; };
|
|
|
|
|
|
2021-06-19 12:10:43 +02:00
|
|
|
void set_close_button_enabled(bool close_button_enabled) { m_close_button_enabled = close_button_enabled; };
|
2022-05-05 14:08:33 +01:00
|
|
|
bool close_button_enabled() const { return m_close_button_enabled; }
|
2021-06-19 12:10:43 +02:00
|
|
|
|
2021-10-07 01:52:04 -06:00
|
|
|
void set_reorder_allowed(bool reorder_allowed) { m_reorder_allowed = reorder_allowed; }
|
|
|
|
|
bool reorder_allowed() const { return m_reorder_allowed; }
|
|
|
|
|
|
2021-04-09 22:23:14 +02:00
|
|
|
Function<void(size_t)> on_tab_count_change;
|
2020-04-23 21:13:47 +02:00
|
|
|
Function<void(Widget&)> on_change;
|
2020-05-06 23:03:03 -04:00
|
|
|
Function<void(Widget&)> on_middle_click;
|
2021-06-19 12:10:43 +02:00
|
|
|
Function<void(Widget&)> on_tab_close_click;
|
2022-04-01 20:58:27 +03:00
|
|
|
Function<void(Widget&, ContextMenuEvent const&)> on_context_menu_request;
|
2021-11-30 09:42:13 -07:00
|
|
|
Function<void(Widget&)> on_double_click;
|
2020-04-06 01:02:24 +03:00
|
|
|
|
2019-05-05 01:31:02 +02:00
|
|
|
protected:
|
2020-02-23 12:07:13 +01:00
|
|
|
TabWidget();
|
|
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
virtual void paint_event(PaintEvent&) override;
|
2020-02-02 12:34:39 +01:00
|
|
|
virtual void child_event(Core::ChildEvent&) override;
|
2020-02-02 15:07:41 +01:00
|
|
|
virtual void resize_event(ResizeEvent&) override;
|
|
|
|
|
virtual void mousedown_event(MouseEvent&) override;
|
2021-06-19 12:10:43 +02:00
|
|
|
virtual void mouseup_event(MouseEvent&) override;
|
2020-02-02 15:07:41 +01:00
|
|
|
virtual void mousemove_event(MouseEvent&) override;
|
2020-02-02 12:34:39 +01:00
|
|
|
virtual void leave_event(Core::Event&) override;
|
2020-04-30 09:04:03 +02:00
|
|
|
virtual void keydown_event(KeyEvent&) override;
|
2020-05-18 18:36:02 -04:00
|
|
|
virtual void context_menu_event(ContextMenuEvent&) override;
|
2021-11-30 09:42:13 -07:00
|
|
|
virtual void doubleclick_event(MouseEvent&) override;
|
2019-05-05 01:31:02 +02:00
|
|
|
|
|
|
|
|
private:
|
2022-04-01 20:58:27 +03:00
|
|
|
Gfx::IntRect child_rect_for_size(Gfx::IntSize const&) const;
|
2021-07-28 01:11:41 +01:00
|
|
|
Gfx::IntRect button_rect(size_t index) const;
|
2022-05-17 20:47:35 +10:00
|
|
|
Gfx::IntRect vertical_button_rect(size_t index) const;
|
|
|
|
|
Gfx::IntRect horizontal_button_rect(size_t index) const;
|
2021-07-28 01:11:41 +01:00
|
|
|
Gfx::IntRect close_button_rect(size_t index) const;
|
2020-06-10 10:57:59 +02:00
|
|
|
Gfx::IntRect bar_rect() const;
|
|
|
|
|
Gfx::IntRect container_rect() const;
|
2019-05-07 14:01:20 +02:00
|
|
|
void update_bar();
|
2020-10-30 16:30:33 +01:00
|
|
|
void update_focus_policy();
|
2021-04-09 22:30:41 +02:00
|
|
|
int bar_margin() const { return 2; }
|
2019-05-05 01:31:02 +02:00
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
RefPtr<Widget> m_active_widget;
|
2019-05-05 01:31:02 +02:00
|
|
|
|
|
|
|
|
struct TabData {
|
2022-04-01 20:58:27 +03:00
|
|
|
int width(Gfx::Font const&) const;
|
2019-05-05 01:31:02 +02:00
|
|
|
String title;
|
2020-04-24 22:27:46 +02:00
|
|
|
RefPtr<Gfx::Bitmap> icon;
|
2020-02-02 15:07:41 +01:00
|
|
|
Widget* widget { nullptr };
|
2019-05-05 01:31:02 +02:00
|
|
|
};
|
|
|
|
|
Vector<TabData> m_tabs;
|
2019-07-28 17:39:04 +02:00
|
|
|
TabPosition m_tab_position { TabPosition::Top };
|
2021-07-28 01:11:41 +01:00
|
|
|
Optional<size_t> m_hovered_tab_index;
|
|
|
|
|
Optional<size_t> m_hovered_close_button_index;
|
|
|
|
|
Optional<size_t> m_pressed_close_button_index;
|
2021-06-15 14:28:39 +02:00
|
|
|
GUI::Margins m_container_margins { 2, 2, 2, 2 };
|
2020-04-24 22:27:46 +02:00
|
|
|
Gfx::TextAlignment m_text_alignment { Gfx::TextAlignment::Center };
|
2020-04-24 22:36:25 +02:00
|
|
|
bool m_uniform_tabs { false };
|
2020-05-18 22:44:08 +01:00
|
|
|
bool m_bar_visible { true };
|
2021-06-19 12:10:43 +02:00
|
|
|
bool m_close_button_enabled { false };
|
2021-10-07 01:52:04 -06:00
|
|
|
|
2022-05-17 20:47:35 +10:00
|
|
|
int m_max_tab_width { 160 };
|
|
|
|
|
int m_min_tab_width { 24 };
|
|
|
|
|
|
2021-10-07 01:52:04 -06:00
|
|
|
bool m_reorder_allowed { false };
|
|
|
|
|
bool m_dragging_active_tab { false };
|
|
|
|
|
int m_grab_offset { 0 };
|
2022-05-17 20:47:35 +10:00
|
|
|
int m_mouse_pos { 0 };
|
2021-10-07 01:52:04 -06:00
|
|
|
|
|
|
|
|
void drag_tab(size_t index);
|
|
|
|
|
void recalculate_tab_order();
|
2019-05-05 01:31:02 +02:00
|
|
|
};
|
2020-02-02 15:07:41 +01:00
|
|
|
|
|
|
|
|
}
|