forked from qt-creator/qt-creator
When resizing the Qt Creator main window to a smaller size, at some point the mode tab icons will shrink to fit all tabs into the window. The icon size will if necessary shrink to tiny sizes below 8px, which is pointless. Let's add some code that hides the icons when a tab becomes smaller than a certain size. Task-Number: QTCREATORBUG-7879 Change-Id: I590379a2ec52a8a86e998c08b29d54aca800e542 Reviewed-by: Thomas Hartmann <Thomas.Hartmann@nokia.com>
530 lines
15 KiB
C++
530 lines
15 KiB
C++
/**************************************************************************
|
|
**
|
|
** This file is part of Qt Creator
|
|
**
|
|
** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies).
|
|
**
|
|
** Contact: http://www.qt-project.org/
|
|
**
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
**
|
|
** This file may be used under the terms of the GNU Lesser General Public
|
|
** License version 2.1 as published by the Free Software Foundation and
|
|
** appearing in the file LICENSE.LGPL included in the packaging of this file.
|
|
** Please review the following information to ensure the GNU Lesser General
|
|
** Public License version 2.1 requirements will be met:
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** Other Usage
|
|
**
|
|
** Alternatively, this file may be used in accordance with the terms and
|
|
** conditions contained in a signed written agreement between you and Nokia.
|
|
**
|
|
**
|
|
**************************************************************************/
|
|
|
|
#include "fancytabwidget.h"
|
|
#include <utils/hostosinfo.h>
|
|
#include <utils/stylehelper.h>
|
|
#include <utils/styledbar.h>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QColorDialog>
|
|
#include <QHBoxLayout>
|
|
#include <QVBoxLayout>
|
|
#include <QMouseEvent>
|
|
#include <QWindowsStyle>
|
|
#include <QPainter>
|
|
#include <QSplitter>
|
|
#include <QStackedLayout>
|
|
#include <QStatusBar>
|
|
#include <QToolButton>
|
|
#include <QToolTip>
|
|
#include <QAnimationGroup>
|
|
#include <QPropertyAnimation>
|
|
|
|
using namespace Core;
|
|
using namespace Internal;
|
|
|
|
const int FancyTabBar::m_rounding = 22;
|
|
const int FancyTabBar::m_textPadding = 4;
|
|
|
|
void FancyTab::fadeIn()
|
|
{
|
|
animator.stop();
|
|
animator.setDuration(80);
|
|
animator.setEndValue(40);
|
|
animator.start();
|
|
}
|
|
|
|
void FancyTab::fadeOut()
|
|
{
|
|
animator.stop();
|
|
animator.setDuration(160);
|
|
animator.setEndValue(0);
|
|
animator.start();
|
|
}
|
|
|
|
void FancyTab::setFader(float value)
|
|
{
|
|
m_fader = value;
|
|
tabbar->update();
|
|
}
|
|
|
|
FancyTabBar::FancyTabBar(QWidget *parent)
|
|
: QWidget(parent)
|
|
{
|
|
m_hoverIndex = -1;
|
|
m_currentIndex = -1;
|
|
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
|
|
setStyle(new QWindowsStyle);
|
|
setMinimumWidth(qMax(2 * m_rounding, 40));
|
|
setAttribute(Qt::WA_Hover, true);
|
|
setFocusPolicy(Qt::NoFocus);
|
|
setMouseTracking(true); // Needed for hover events
|
|
m_triggerTimer.setSingleShot(true);
|
|
|
|
// We use a zerotimer to keep the sidebar responsive
|
|
connect(&m_triggerTimer, SIGNAL(timeout()), this, SLOT(emitCurrentIndex()));
|
|
}
|
|
|
|
FancyTabBar::~FancyTabBar()
|
|
{
|
|
delete style();
|
|
}
|
|
|
|
QSize FancyTabBar::tabSizeHint(bool minimum) const
|
|
{
|
|
QFont boldFont(font());
|
|
boldFont.setPointSizeF(Utils::StyleHelper::sidebarFontSize());
|
|
boldFont.setBold(true);
|
|
QFontMetrics fm(boldFont);
|
|
int spacing = 8;
|
|
int width = 60 + spacing + 2;
|
|
int maxLabelwidth = 0;
|
|
for (int tab=0 ; tab<count() ;++tab) {
|
|
int width = fm.width(tabText(tab));
|
|
if (width > maxLabelwidth)
|
|
maxLabelwidth = width;
|
|
}
|
|
int iconHeight = minimum ? 0 : 32;
|
|
return QSize(qMax(width, maxLabelwidth + 4), iconHeight + spacing + fm.height());
|
|
}
|
|
|
|
void FancyTabBar::paintEvent(QPaintEvent *event)
|
|
{
|
|
Q_UNUSED(event)
|
|
QPainter p(this);
|
|
|
|
for (int i = 0; i < count(); ++i)
|
|
if (i != currentIndex())
|
|
paintTab(&p, i);
|
|
|
|
// paint active tab last, since it overlaps the neighbors
|
|
if (currentIndex() != -1)
|
|
paintTab(&p, currentIndex());
|
|
}
|
|
|
|
// Handle hover events for mouse fade ins
|
|
void FancyTabBar::mouseMoveEvent(QMouseEvent *e)
|
|
{
|
|
int newHover = -1;
|
|
for (int i = 0; i < count(); ++i) {
|
|
QRect area = tabRect(i);
|
|
if (area.contains(e->pos())) {
|
|
newHover = i;
|
|
break;
|
|
}
|
|
}
|
|
if (newHover == m_hoverIndex)
|
|
return;
|
|
|
|
if (validIndex(m_hoverIndex))
|
|
m_tabs[m_hoverIndex]->fadeOut();
|
|
|
|
m_hoverIndex = newHover;
|
|
|
|
if (validIndex(m_hoverIndex)) {
|
|
m_tabs[m_hoverIndex]->fadeIn();
|
|
m_hoverRect = tabRect(m_hoverIndex);
|
|
}
|
|
}
|
|
|
|
bool FancyTabBar::event(QEvent *event)
|
|
{
|
|
if (event->type() == QEvent::ToolTip) {
|
|
if (validIndex(m_hoverIndex)) {
|
|
QString tt = tabToolTip(m_hoverIndex);
|
|
if (!tt.isEmpty()) {
|
|
QToolTip::showText(static_cast<QHelpEvent*>(event)->globalPos(), tt, this);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return QWidget::event(event);
|
|
}
|
|
|
|
// Resets hover animation on mouse enter
|
|
void FancyTabBar::enterEvent(QEvent *e)
|
|
{
|
|
Q_UNUSED(e)
|
|
m_hoverRect = QRect();
|
|
m_hoverIndex = -1;
|
|
}
|
|
|
|
// Resets hover animation on mouse enter
|
|
void FancyTabBar::leaveEvent(QEvent *e)
|
|
{
|
|
Q_UNUSED(e)
|
|
m_hoverIndex = -1;
|
|
m_hoverRect = QRect();
|
|
for (int i = 0 ; i < m_tabs.count() ; ++i) {
|
|
m_tabs[i]->fadeOut();
|
|
}
|
|
}
|
|
|
|
QSize FancyTabBar::sizeHint() const
|
|
{
|
|
QSize sh = tabSizeHint();
|
|
return QSize(sh.width(), sh.height() * m_tabs.count());
|
|
}
|
|
|
|
QSize FancyTabBar::minimumSizeHint() const
|
|
{
|
|
QSize sh = tabSizeHint(true);
|
|
return QSize(sh.width(), sh.height() * m_tabs.count());
|
|
}
|
|
|
|
QRect FancyTabBar::tabRect(int index) const
|
|
{
|
|
QSize sh = tabSizeHint();
|
|
|
|
if (sh.height() * m_tabs.count() > height())
|
|
sh.setHeight(height() / m_tabs.count());
|
|
|
|
return QRect(0, index * sh.height(), sh.width(), sh.height());
|
|
|
|
}
|
|
|
|
// This keeps the sidebar responsive since
|
|
// we get a repaint before loading the
|
|
// mode itself
|
|
void FancyTabBar::emitCurrentIndex()
|
|
{
|
|
emit currentChanged(m_currentIndex);
|
|
}
|
|
|
|
void FancyTabBar::mousePressEvent(QMouseEvent *e)
|
|
{
|
|
e->accept();
|
|
for (int index = 0; index < m_tabs.count(); ++index) {
|
|
if (tabRect(index).contains(e->pos())) {
|
|
|
|
if (isTabEnabled(index)) {
|
|
m_currentIndex = index;
|
|
update();
|
|
m_triggerTimer.start(0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void FancyTabBar::paintTab(QPainter *painter, int tabIndex) const
|
|
{
|
|
if (!validIndex(tabIndex)) {
|
|
qWarning("invalid index");
|
|
return;
|
|
}
|
|
painter->save();
|
|
|
|
QRect rect = tabRect(tabIndex);
|
|
bool selected = (tabIndex == m_currentIndex);
|
|
bool enabled = isTabEnabled(tabIndex);
|
|
|
|
if (selected) {
|
|
//background
|
|
painter->save();
|
|
QLinearGradient grad(rect.topLeft(), rect.topRight());
|
|
grad.setColorAt(0, QColor(255, 255, 255, 140));
|
|
grad.setColorAt(1, QColor(255, 255, 255, 210));
|
|
painter->fillRect(rect.adjusted(0, 0, 0, -1), grad);
|
|
painter->restore();
|
|
|
|
//shadows
|
|
painter->setPen(QColor(0, 0, 0, 110));
|
|
painter->drawLine(rect.topLeft() + QPoint(1,-1), rect.topRight() - QPoint(0,1));
|
|
painter->drawLine(rect.bottomLeft(), rect.bottomRight());
|
|
painter->setPen(QColor(0, 0, 0, 40));
|
|
painter->drawLine(rect.topLeft(), rect.bottomLeft());
|
|
|
|
//highlights
|
|
painter->setPen(QColor(255, 255, 255, 50));
|
|
painter->drawLine(rect.topLeft() + QPoint(0, -2), rect.topRight() - QPoint(0,2));
|
|
painter->drawLine(rect.bottomLeft() + QPoint(0, 1), rect.bottomRight() + QPoint(0,1));
|
|
painter->setPen(QColor(255, 255, 255, 40));
|
|
painter->drawLine(rect.topLeft() + QPoint(0, 0), rect.topRight());
|
|
painter->drawLine(rect.topRight() + QPoint(0, 1), rect.bottomRight() - QPoint(0, 1));
|
|
painter->drawLine(rect.bottomLeft() + QPoint(0,-1), rect.bottomRight()-QPoint(0,1));
|
|
}
|
|
|
|
QString tabText(this->tabText(tabIndex));
|
|
QRect tabTextRect(rect);
|
|
const bool drawIcon = rect.height() > 36;
|
|
QRect tabIconRect(tabTextRect);
|
|
tabTextRect.translate(0, drawIcon ? -2 : 1);
|
|
QFont boldFont(painter->font());
|
|
boldFont.setPointSizeF(Utils::StyleHelper::sidebarFontSize());
|
|
boldFont.setBold(true);
|
|
painter->setFont(boldFont);
|
|
painter->setPen(selected ? QColor(255, 255, 255, 160) : QColor(0, 0, 0, 110));
|
|
const int textFlags = Qt::AlignCenter | (drawIcon ? Qt::AlignBottom : Qt::AlignVCenter) | Qt::TextWordWrap;
|
|
if (enabled) {
|
|
painter->drawText(tabTextRect, textFlags, tabText);
|
|
painter->setPen(selected ? QColor(60, 60, 60) : Utils::StyleHelper::panelTextColor());
|
|
} else {
|
|
painter->setPen(selected ? Utils::StyleHelper::panelTextColor() : QColor(255, 255, 255, 120));
|
|
}
|
|
if (!Utils::HostOsInfo::isMacHost() && !selected && enabled) {
|
|
painter->save();
|
|
int fader = int(m_tabs[tabIndex]->fader());
|
|
QLinearGradient grad(rect.topLeft(), rect.topRight());
|
|
grad.setColorAt(0, Qt::transparent);
|
|
grad.setColorAt(0.5, QColor(255, 255, 255, fader));
|
|
grad.setColorAt(1, Qt::transparent);
|
|
painter->fillRect(rect, grad);
|
|
painter->setPen(QPen(grad, 1.0));
|
|
painter->drawLine(rect.topLeft(), rect.topRight());
|
|
painter->drawLine(rect.bottomLeft(), rect.bottomRight());
|
|
painter->restore();
|
|
}
|
|
|
|
if (!enabled)
|
|
painter->setOpacity(0.7);
|
|
|
|
if (drawIcon) {
|
|
int textHeight = painter->fontMetrics().boundingRect(QRect(0, 0, width(), height()), Qt::TextWordWrap, tabText).height();
|
|
tabIconRect.adjust(0, 4, 0, -textHeight);
|
|
Utils::StyleHelper::drawIconWithShadow(tabIcon(tabIndex), tabIconRect, painter, enabled ? QIcon::Normal : QIcon::Disabled);
|
|
}
|
|
|
|
painter->translate(0, -1);
|
|
painter->drawText(tabTextRect, textFlags, tabText);
|
|
painter->restore();
|
|
}
|
|
|
|
void FancyTabBar::setCurrentIndex(int index) {
|
|
if (isTabEnabled(index)) {
|
|
m_currentIndex = index;
|
|
update();
|
|
emit currentChanged(m_currentIndex);
|
|
}
|
|
}
|
|
|
|
void FancyTabBar::setTabEnabled(int index, bool enable)
|
|
{
|
|
Q_ASSERT(index < m_tabs.size());
|
|
Q_ASSERT(index >= 0);
|
|
|
|
if (index < m_tabs.size() && index >= 0) {
|
|
m_tabs[index]->enabled = enable;
|
|
update(tabRect(index));
|
|
}
|
|
}
|
|
|
|
bool FancyTabBar::isTabEnabled(int index) const
|
|
{
|
|
Q_ASSERT(index < m_tabs.size());
|
|
Q_ASSERT(index >= 0);
|
|
|
|
if (index < m_tabs.size() && index >= 0)
|
|
return m_tabs[index]->enabled;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
//////
|
|
// FancyColorButton
|
|
//////
|
|
|
|
class FancyColorButton : public QWidget
|
|
{
|
|
public:
|
|
FancyColorButton(QWidget *parent)
|
|
: m_parent(parent)
|
|
{
|
|
setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Preferred);
|
|
}
|
|
|
|
void mousePressEvent(QMouseEvent *ev)
|
|
{
|
|
if (ev->modifiers() & Qt::ShiftModifier) {
|
|
QColor color = QColorDialog::getColor(Utils::StyleHelper::requestedBaseColor(), m_parent);
|
|
if (color.isValid())
|
|
Utils::StyleHelper::setBaseColor(color);
|
|
}
|
|
}
|
|
private:
|
|
QWidget *m_parent;
|
|
};
|
|
|
|
//////
|
|
// FancyTabWidget
|
|
//////
|
|
|
|
FancyTabWidget::FancyTabWidget(QWidget *parent)
|
|
: QWidget(parent)
|
|
{
|
|
m_tabBar = new FancyTabBar(this);
|
|
|
|
m_selectionWidget = new QWidget(this);
|
|
QVBoxLayout *selectionLayout = new QVBoxLayout;
|
|
selectionLayout->setSpacing(0);
|
|
selectionLayout->setMargin(0);
|
|
|
|
Utils::StyledBar *bar = new Utils::StyledBar;
|
|
QHBoxLayout *layout = new QHBoxLayout(bar);
|
|
layout->setMargin(0);
|
|
layout->setSpacing(0);
|
|
layout->addWidget(new FancyColorButton(this));
|
|
selectionLayout->addWidget(bar);
|
|
|
|
selectionLayout->addWidget(m_tabBar, 1);
|
|
m_selectionWidget->setLayout(selectionLayout);
|
|
m_selectionWidget->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding);
|
|
|
|
m_cornerWidgetContainer = new QWidget(this);
|
|
m_cornerWidgetContainer->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Preferred);
|
|
m_cornerWidgetContainer->setAutoFillBackground(false);
|
|
|
|
QVBoxLayout *cornerWidgetLayout = new QVBoxLayout;
|
|
cornerWidgetLayout->setSpacing(0);
|
|
cornerWidgetLayout->setMargin(0);
|
|
cornerWidgetLayout->addStretch();
|
|
m_cornerWidgetContainer->setLayout(cornerWidgetLayout);
|
|
|
|
selectionLayout->addWidget(m_cornerWidgetContainer, 0);
|
|
|
|
m_modesStack = new QStackedLayout;
|
|
m_statusBar = new QStatusBar;
|
|
m_statusBar->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Fixed);
|
|
|
|
QVBoxLayout *vlayout = new QVBoxLayout;
|
|
vlayout->setMargin(0);
|
|
vlayout->setSpacing(0);
|
|
vlayout->addLayout(m_modesStack);
|
|
vlayout->addWidget(m_statusBar);
|
|
|
|
QHBoxLayout *mainLayout = new QHBoxLayout;
|
|
mainLayout->setMargin(0);
|
|
mainLayout->setSpacing(1);
|
|
mainLayout->addWidget(m_selectionWidget);
|
|
mainLayout->addLayout(vlayout);
|
|
setLayout(mainLayout);
|
|
|
|
connect(m_tabBar, SIGNAL(currentChanged(int)), this, SLOT(showWidget(int)));
|
|
}
|
|
|
|
void FancyTabWidget::setSelectionWidgetHidden(bool hidden) {
|
|
m_selectionWidget->setHidden(hidden);
|
|
}
|
|
|
|
void FancyTabWidget::insertTab(int index, QWidget *tab, const QIcon &icon, const QString &label)
|
|
{
|
|
m_modesStack->insertWidget(index, tab);
|
|
m_tabBar->insertTab(index, icon, label);
|
|
}
|
|
|
|
void FancyTabWidget::removeTab(int index)
|
|
{
|
|
m_modesStack->removeWidget(m_modesStack->widget(index));
|
|
m_tabBar->removeTab(index);
|
|
}
|
|
|
|
void FancyTabWidget::setBackgroundBrush(const QBrush &brush)
|
|
{
|
|
QPalette pal = m_tabBar->palette();
|
|
pal.setBrush(QPalette::Mid, brush);
|
|
m_tabBar->setPalette(pal);
|
|
pal = m_cornerWidgetContainer->palette();
|
|
pal.setBrush(QPalette::Mid, brush);
|
|
m_cornerWidgetContainer->setPalette(pal);
|
|
}
|
|
|
|
void FancyTabWidget::paintEvent(QPaintEvent *event)
|
|
{
|
|
Q_UNUSED(event)
|
|
QPainter painter(this);
|
|
|
|
QRect rect = m_selectionWidget->rect().adjusted(0, 0, 1, 0);
|
|
rect = style()->visualRect(layoutDirection(), geometry(), rect);
|
|
Utils::StyleHelper::verticalGradient(&painter, rect, rect);
|
|
painter.setPen(Utils::StyleHelper::borderColor());
|
|
painter.drawLine(rect.topRight(), rect.bottomRight());
|
|
|
|
QColor light = Utils::StyleHelper::sidebarHighlight();
|
|
painter.setPen(light);
|
|
painter.drawLine(rect.bottomLeft(), rect.bottomRight());
|
|
}
|
|
|
|
void FancyTabWidget::insertCornerWidget(int pos, QWidget *widget)
|
|
{
|
|
QVBoxLayout *layout = static_cast<QVBoxLayout *>(m_cornerWidgetContainer->layout());
|
|
layout->insertWidget(pos, widget);
|
|
}
|
|
|
|
int FancyTabWidget::cornerWidgetCount() const
|
|
{
|
|
return m_cornerWidgetContainer->layout()->count();
|
|
}
|
|
|
|
void FancyTabWidget::addCornerWidget(QWidget *widget)
|
|
{
|
|
m_cornerWidgetContainer->layout()->addWidget(widget);
|
|
}
|
|
|
|
int FancyTabWidget::currentIndex() const
|
|
{
|
|
return m_tabBar->currentIndex();
|
|
}
|
|
|
|
QStatusBar *FancyTabWidget::statusBar() const
|
|
{
|
|
return m_statusBar;
|
|
}
|
|
|
|
void FancyTabWidget::setCurrentIndex(int index)
|
|
{
|
|
if (m_tabBar->isTabEnabled(index))
|
|
m_tabBar->setCurrentIndex(index);
|
|
}
|
|
|
|
void FancyTabWidget::showWidget(int index)
|
|
{
|
|
emit currentAboutToShow(index);
|
|
m_modesStack->setCurrentIndex(index);
|
|
emit currentChanged(index);
|
|
}
|
|
|
|
void FancyTabWidget::setTabToolTip(int index, const QString &toolTip)
|
|
{
|
|
m_tabBar->setTabToolTip(index, toolTip);
|
|
}
|
|
|
|
void FancyTabWidget::setTabEnabled(int index, bool enable)
|
|
{
|
|
m_tabBar->setTabEnabled(index, enable);
|
|
}
|
|
|
|
bool FancyTabWidget::isTabEnabled(int index) const
|
|
{
|
|
return m_tabBar->isTabEnabled(index);
|
|
}
|