/**************************************************************************** ** ** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, 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, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "fancytabwidget.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include 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(QStyleFactory::create(QLatin1String("windows"))); 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 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(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(tabRect(tabIndex)); QRect tabIconRect(tabTextRect); tabTextRect.translate(0, -2); 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)); int textFlags = Qt::AlignCenter | Qt::AlignBottom | 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)); } #ifndef Q_OS_MAC if (!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(); } #endif if (!enabled) painter->setOpacity(0.7); 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(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); }