2024-04-03 01:24:17 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2024, Tim Flynn <trflynn89@serenityos.org>
|
2024-04-28 14:07:23 +00:00
|
|
|
* Copyright (c) 2024, Jamie Mansfield <jmansfield@cadixdev.org>
|
2024-07-26 10:38:26 +00:00
|
|
|
* Copyright (c) 2024, Sam Atkins <sam@ladybird.org>
|
2024-04-03 01:24:17 +00:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2024-04-28 14:07:23 +00:00
|
|
|
#include "Tab.h"
|
2024-04-03 01:38:01 +00:00
|
|
|
#include <AK/StdLibExtras.h>
|
2024-04-28 14:07:23 +00:00
|
|
|
#include <AK/TypeCasts.h>
|
2024-04-03 01:24:17 +00:00
|
|
|
#include <Ladybird/Qt/TabBar.h>
|
2024-04-28 14:07:23 +00:00
|
|
|
#include <QContextMenuEvent>
|
2024-04-03 01:24:17 +00:00
|
|
|
#include <QEvent>
|
|
|
|
#include <QPushButton>
|
2024-07-26 10:38:26 +00:00
|
|
|
#include <QStylePainter>
|
2024-04-03 01:24:17 +00:00
|
|
|
|
|
|
|
namespace Ladybird {
|
|
|
|
|
2024-04-28 14:07:23 +00:00
|
|
|
TabBar::TabBar(QWidget* parent)
|
|
|
|
: QTabBar(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-04-03 01:38:01 +00:00
|
|
|
QSize TabBar::tabSizeHint(int index) const
|
|
|
|
{
|
|
|
|
auto width = this->width() / count();
|
|
|
|
width = min(225, width);
|
2024-04-28 11:12:47 +00:00
|
|
|
width = max(128, width);
|
2024-04-03 01:38:01 +00:00
|
|
|
|
|
|
|
auto hint = QTabBar::tabSizeHint(index);
|
|
|
|
hint.setWidth(width);
|
|
|
|
return hint;
|
|
|
|
}
|
|
|
|
|
2024-04-28 14:07:23 +00:00
|
|
|
void TabBar::contextMenuEvent(QContextMenuEvent* event)
|
|
|
|
{
|
|
|
|
auto* tab_widget = verify_cast<QTabWidget>(this->parent());
|
|
|
|
auto* tab = verify_cast<Tab>(tab_widget->widget(tabAt(event->pos())));
|
|
|
|
if (tab)
|
|
|
|
tab->context_menu()->exec(event->globalPos());
|
|
|
|
}
|
|
|
|
|
2024-04-03 01:38:01 +00:00
|
|
|
TabWidget::TabWidget(QWidget* parent)
|
|
|
|
: QTabWidget(parent)
|
|
|
|
{
|
|
|
|
// This must be called first, otherwise several of the options below have no effect.
|
2024-04-28 14:07:23 +00:00
|
|
|
setTabBar(new TabBar(this));
|
2024-04-03 01:38:01 +00:00
|
|
|
|
|
|
|
setDocumentMode(true);
|
|
|
|
setElideMode(Qt::TextElideMode::ElideRight);
|
|
|
|
setMovable(true);
|
|
|
|
setTabsClosable(true);
|
|
|
|
|
2024-07-26 10:38:26 +00:00
|
|
|
setStyle(new TabStyle(this));
|
|
|
|
|
2024-04-03 01:38:01 +00:00
|
|
|
installEventFilter(parent);
|
|
|
|
}
|
|
|
|
|
2024-07-26 13:39:26 +00:00
|
|
|
void TabWidget::paintEvent(QPaintEvent*)
|
|
|
|
{
|
|
|
|
auto prepare_style_options = [](QTabBar* tab_bar, QSize widget_size) {
|
|
|
|
QStyleOptionTabBarBase style_options;
|
|
|
|
QStyleOptionTab tab_overlap;
|
|
|
|
tab_overlap.shape = tab_bar->shape();
|
|
|
|
auto overlap = tab_bar->style()->pixelMetric(QStyle::PM_TabBarBaseOverlap, &tab_overlap, tab_bar);
|
|
|
|
style_options.initFrom(tab_bar);
|
|
|
|
style_options.shape = tab_bar->shape();
|
|
|
|
style_options.documentMode = tab_bar->documentMode();
|
|
|
|
// NOTE: This assumes the tab bar is at the top of the tab widget.
|
|
|
|
style_options.rect = { 0, widget_size.height() - overlap, widget_size.width(), overlap };
|
|
|
|
|
|
|
|
return style_options;
|
|
|
|
};
|
|
|
|
|
|
|
|
QStylePainter painter { this, tabBar() };
|
|
|
|
if (auto* widget = cornerWidget(Qt::TopRightCorner)) {
|
|
|
|
// Manually paint the background for the area where the "new tab" button would have been
|
|
|
|
// if we hadn't relocated it in `TabStyle::subElementRect()`.
|
|
|
|
auto style_options = prepare_style_options(tabBar(), widget->size());
|
|
|
|
style_options.rect.translate(tabBar()->rect().width(), widget->y());
|
|
|
|
painter.drawPrimitive(QStyle::PE_FrameTabBarBase, style_options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-03 01:24:17 +00:00
|
|
|
TabBarButton::TabBarButton(QIcon const& icon, QWidget* parent)
|
|
|
|
: QPushButton(icon, {}, parent)
|
|
|
|
{
|
|
|
|
resize({ 20, 20 });
|
|
|
|
setFlat(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TabBarButton::event(QEvent* event)
|
|
|
|
{
|
|
|
|
if (event->type() == QEvent::Enter)
|
|
|
|
setFlat(false);
|
|
|
|
if (event->type() == QEvent::Leave)
|
|
|
|
setFlat(true);
|
|
|
|
|
|
|
|
return QPushButton::event(event);
|
|
|
|
}
|
|
|
|
|
2024-07-26 10:38:26 +00:00
|
|
|
TabStyle::TabStyle(QObject* parent)
|
|
|
|
{
|
|
|
|
setParent(parent);
|
|
|
|
}
|
|
|
|
|
|
|
|
QRect TabStyle::subElementRect(QStyle::SubElement sub_element, QStyleOption const* option, QWidget const* widget) const
|
|
|
|
{
|
|
|
|
// Place our add-tab button (set as the top-right corner widget) directly after the last tab
|
|
|
|
if (sub_element == QStyle::SE_TabWidgetRightCorner) {
|
|
|
|
auto* tab_widget = verify_cast<TabWidget>(widget);
|
|
|
|
auto tab_bar_size = tab_widget->tabBar()->sizeHint();
|
|
|
|
auto new_tab_button_size = tab_bar_size.height();
|
|
|
|
return QRect {
|
|
|
|
qMin(tab_bar_size.width(), tab_widget->width() - new_tab_button_size),
|
|
|
|
0,
|
|
|
|
new_tab_button_size,
|
|
|
|
new_tab_button_size
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return QProxyStyle::subElementRect(sub_element, option, widget);
|
|
|
|
}
|
|
|
|
|
2024-04-03 01:24:17 +00:00
|
|
|
}
|