Files
cantata/support/fancytabwidget.cpp
2014-02-25 19:26:32 +00:00

908 lines
26 KiB
C++

/*
* Cantata
*
* Copyright (c) 2011-2014 Craig Drummond <craig.p.drummond@gmail.com>
*/
/**************************************************************************
**
** This file is part of Qt Creator
**
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
**
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** Commercial Usage
**
** Licensees holding valid Qt Commercial licenses may use this file in
** accordance with the Qt Commercial License Agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Nokia.
**
** 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.
**
** If you are unsure which license is appropriate for your use, please
** contact the sales department at http://qt.nokia.com/contact.
**
**************************************************************************/
#include "fancytabwidget.h"
#include "localize.h"
#include "icon.h"
#include "gtkstyle.h"
#include "action.h"
#include "utils.h"
#include <QHBoxLayout>
#include <QMouseEvent>
#include <QPainter>
#include <QSplitter>
#include <QStackedWidget>
#include <QStyleOption>
#include <QToolButton>
#include <QToolTip>
#include <QVBoxLayout>
#include <QApplication>
#include <QAnimationGroup>
#include <QPropertyAnimation>
#include <QSignalMapper>
#include <QMenu>
static inline int sidebarSpacing(bool withText)
{
int sp=Utils::isHighDpi() ? 24 : 12;
if (!withText) {
sp*=1.25;
}
return sp;
}
static inline Qt::TextElideMode elideMode()
{
return Qt::LeftToRight==QApplication::layoutDirection() ? Qt::ElideRight : Qt::ElideLeft;
}
using namespace Core;
using namespace Internal;
static int largeIconSize=32;
static int smallIconSize=16;
void FancyTabWidget::setup()
{
if (Utils::isHighDpi()) {
largeIconSize=Icon::stdSize(40);
smallIconSize=16;
if (largeIconSize>32) {
if (largeIconSize<56) {
smallIconSize=22;
} else {
smallIconSize=32;
}
}
} else {
largeIconSize=32;
smallIconSize=16;
}
}
int FancyTabWidget::iconSize(bool large)
{
return large ? largeIconSize : smallIconSize;
}
static void drawIcon(const QIcon &icon, const QRect &r, QPainter *p, const QSize &iconSize, bool selected)
{
QPixmap px = icon.pixmap(iconSize, selected ? QIcon::Selected : QIcon::Normal);
p->drawPixmap(r.x()+(r.width()-px.width())/2.0, r.y()+(r.height()-px.height())/2.0, px.width(), px.height(), px);
}
void FancyTabProxyStyle::drawPrimitive(PrimitiveElement element, const QStyleOption *option, QPainter *p, const QWidget *widget) const
{
if (PE_FrameTabBarBase!=element) {
QProxyStyle::drawPrimitive(element, option, p, widget);
}
}
int FancyTabProxyStyle::styleHint(StyleHint hint, const QStyleOption *option, const QWidget *widget, QStyleHintReturn *returnData) const
{
if (SH_TabBar_Alignment==hint && widget && qobject_cast<const QTabBar *>(widget)) {
QTabBar::Shape shape=static_cast<const QTabBar *>(widget)->shape();
if (QTabBar::RoundedNorth==shape || QTabBar::RoundedSouth==shape) {
return (int)Qt::AlignCenter;
}
}
return QProxyStyle::styleHint(hint, option, widget, returnData);
}
void FancyTabProxyStyle::drawControl(ControlElement element, const QStyleOption *option, QPainter *p, const QWidget *widget) const
{
const QStyleOptionTabV3* v_opt = qstyleoption_cast<const QStyleOptionTabV3*>(option);
if (element != CE_TabBarTab || !v_opt) {
QProxyStyle::drawControl(element, option, p, widget);
return;
}
const QRect rect = v_opt->rect;
const bool selected = v_opt->state & State_Selected;
const bool vertical_tabs = v_opt->shape == QTabBar::RoundedWest;
const QString text = v_opt->text;
QTransform m;
if (vertical_tabs) {
m = QTransform::fromTranslate(rect.left(), rect.bottom());
m.rotate(-90);
} else {
m = QTransform::fromTranslate(rect.left(), rect.top());
}
const QRect draw_rect(QPoint(0, 0), m.mapRect(rect).size());
if (!selected && GtkStyle::isActive()) {
p->fillRect(option->rect, option->palette.background());
}
p->save();
p->setTransform(m);
QRect icon_rect(QPoint(8, 0), v_opt->iconSize);
QRect text_rect(icon_rect.topRight() + QPoint(4, 0), draw_rect.size());
text_rect.setRight(draw_rect.width());
icon_rect.translate(0, (draw_rect.height() - icon_rect.height()) / 2);
QStyleOptionViewItemV4 styleOpt;
styleOpt.palette=option->palette;
styleOpt.rect=draw_rect;
if (QStyleOptionTab::Beginning==v_opt->position) {
styleOpt.rect.adjust(0, 0, -1, 0);
}
styleOpt.state=option->state;
styleOpt.state&=~(QStyle::State_Selected|QStyle::State_MouseOver);
styleOpt.state|=QStyle::State_Selected|QStyle::State_Enabled;
styleOpt.viewItemPosition = QStyleOptionViewItemV4::OnlyOne;
styleOpt.showDecorationSelected=true;
bool drawBgnd=true;
int fader = 1;
if (!selected && drawBgnd) {
const QString fader_key = "tab_" + text + "_fader";
const QString animation_key = "tab_" + text + "_animation";
const QString tab_hover = widget->property("tab_hover").toString();
fader=widget->property(fader_key.toUtf8().constData()).toInt();
QPropertyAnimation* animation = widget->property(animation_key.toUtf8().constData()).value<QPropertyAnimation*>();
if (!animation) {
QWidget* mut_widget = const_cast<QWidget*>(widget);
fader = 0;
mut_widget->setProperty(fader_key.toUtf8().constData(), fader);
animation = new QPropertyAnimation(mut_widget, fader_key.toUtf8(), mut_widget);
connect(animation, SIGNAL(valueChanged(QVariant)), mut_widget, SLOT(update()));
mut_widget->setProperty(animation_key.toUtf8().constData(), QVariant::fromValue(animation));
}
if (text == tab_hover) {
if (animation->state() != QAbstractAnimation::Running && fader != 40) {
animation->stop();
animation->setDuration(80);
animation->setEndValue(50);
animation->start();
}
} else {
if (animation->state() != QAbstractAnimation::Running && fader != 0) {
animation->stop();
animation->setDuration(160);
animation->setEndValue(0);
animation->start();
}
}
if (fader<1) {
drawBgnd=false;
} else {
QColor col(styleOpt.palette.highlight().color());
col.setAlpha(fader);
styleOpt.palette.setColor(styleOpt.palette.currentColorGroup(), QPalette::Highlight, col);
}
}
if (drawBgnd) {
if (!selected && GtkStyle::isActive()) {
GtkStyle::drawSelection(styleOpt, p, (fader*1.0)/150.0);
} else {
QApplication::style()->drawPrimitive(QStyle::PE_PanelItemViewItem, &styleOpt, p, 0);
}
}
int textFlags = Qt::AlignTop | Qt::AlignVCenter;
p->setPen(selected && option->state&State_Active
? QApplication::palette().highlightedText().color() : QApplication::palette().foreground().color());
drawIcon(v_opt->icon, icon_rect, p, v_opt->iconSize,
selected && option->state&State_Active);
QString txt=text;
txt.replace("&", "");
txt=p->fontMetrics().elidedText(txt, elideMode(), text_rect.width());
p->drawText(text_rect.translated(0, -1), textFlags, txt);
p->restore();
}
void FancyTabProxyStyle::polish(QWidget* widget)
{
if ("QTabBar"==QString(widget->metaObject()->className())) {
widget->setMouseTracking(true);
widget->installEventFilter(this);
}
QProxyStyle::polish(widget);
}
void FancyTabProxyStyle::polish(QApplication* app)
{
QProxyStyle::polish(app);
}
void FancyTabProxyStyle::polish(QPalette& palette)
{
QProxyStyle::polish(palette);
}
bool FancyTabProxyStyle::eventFilter(QObject* o, QEvent* e)
{
QTabBar* bar = qobject_cast<QTabBar*>(o);
if (bar && (e->type() == QEvent::MouseMove || e->type() == QEvent::Leave)) {
QMouseEvent *event = static_cast<QMouseEvent*>(e);
const QString old_hovered_tab = bar->property("tab_hover").toString();
const QString hovered_tab = e->type() == QEvent::Leave ? QString() : bar->tabText(bar->tabAt(event->pos()));
bar->setProperty("tab_hover", hovered_tab);
if (old_hovered_tab != hovered_tab) {
bar->update();
}
}
return false;
}
FancyTab::FancyTab(FancyTabBar* tabbar)
: QWidget(tabbar), tabbar(tabbar), m_fader(0)
{
animator.setPropertyName("fader");
animator.setTargetObject(this);
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Maximum);
}
void FancyTab::fadeIn()
{
animator.stop();
animator.setDuration(80);
animator.setEndValue(50);
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, bool text, int iSize, Pos pos)
: QWidget(parent)
, m_showText(text)
, m_pos(pos)
, m_iconSize(iSize)
{
QFont f(font());
if (f.pointSizeF()>=8) {
f.setPointSizeF(f.pointSizeF()*0.8);
setFont(f);
}
setAttribute(Qt::WA_Hover, true);
setFocusPolicy(Qt::NoFocus);
setMouseTracking(true); // Needed for hover events
m_triggerTimer.setSingleShot(true);
QBoxLayout* layout=0;
if (Side!=m_pos) {
setMinimumHeight(tabSizeHint().height());
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
layout=new QHBoxLayout;
} else {
setMinimumWidth(tabSizeHint().width());
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
layout=new QVBoxLayout;
layout->addSpacerItem(new QSpacerItem(0, 0, QSizePolicy::Fixed, QSizePolicy::MinimumExpanding));
}
layout->setSpacing(0);
layout->setContentsMargins(0, 0, 0, 0);
setLayout(layout);
// We use a zerotimer to keep the sidebar responsive
connect(&m_triggerTimer, SIGNAL(timeout()), this, SLOT(emitCurrentIndex()));
}
FancyTabBar::~FancyTabBar()
{
}
QSize FancyTab::sizeHint() const
{
int iconSize=tabbar->iconSize();
bool withText=tabbar->showText();
int spacing = sidebarSpacing(withText);
if (withText) {
QFontMetrics fm(font());
int textWidth = fm.width(text)*1.1;
int width = qMax(iconSize, qMin(3*iconSize, textWidth)) + spacing;
return QSize(width, iconSize + spacing + fm.height());
} else {
return QSize(iconSize + spacing, iconSize + spacing);
}
}
QSize FancyTabBar::tabSizeHint() const
{
int spacing = sidebarSpacing(m_showText);
if (m_showText) {
QFontMetrics fm(font());
int maxTw=0;
foreach (FancyTab *tab, m_tabs) {
maxTw=qMax(maxTw, tab->sizeHint().width());
}
return QSize(qMax(m_iconSize + spacing, maxTw), m_iconSize + spacing + fm.height());
} else {
return QSize(m_iconSize + spacing, m_iconSize + spacing);
}
}
void FancyTabBar::paintEvent(QPaintEvent *event)
{
Q_UNUSED(event)
QPainter p(this);
bool gtkStyle=GtkStyle::isActive();
for (int i = 0; i < count(); ++i) {
if (i != currentIndex()) {
paintTab(&p, i, gtkStyle);
}
}
// paint active tab last, since it overlaps the neighbors
if (currentIndex() != -1) {
paintTab(&p, currentIndex(), gtkStyle);
}
}
void FancyTab::enterEvent(QEvent*)
{
fadeIn();
}
void FancyTab::leaveEvent(QEvent*)
{
fadeOut();
}
QSize FancyTabBar::sizeHint() const
{
QSize sh = tabSizeHint();
return Side!=m_pos
? QSize(sh.width() * m_tabs.count(), sh.height())
: QSize(sh.width(), sh.height() * m_tabs.count());
}
QSize FancyTabBar::minimumSizeHint() const
{
QSize sh = tabSizeHint();
return Side!=m_pos
? QSize(sh.width() * m_tabs.count(), sh.height())
: QSize(sh.width(), sh.height() * m_tabs.count());
}
QRect FancyTabBar::tabRect(int index) const
{
return m_tabs[index]->geometry();
}
QString FancyTabBar::tabToolTip(int index) const
{
return m_tabs[index]->toolTip();
}
void FancyTabBar::setTabToolTip(int index, const QString& toolTip)
{
m_tabs[index]->setToolTip(toolTip);
}
// 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)
{
if (Qt::LeftButton!=e->button()) {
return;
}
e->accept();
for (int index = 0; index < m_tabs.count(); ++index) {
if (tabRect(index).contains(e->pos())) {
m_currentIndex = index;
update();
m_triggerTimer.start(0);
break;
}
}
}
void FancyTabBar::addTab(const QIcon& icon, const QString& label, const QString &tt)
{
FancyTab *tab = new FancyTab(this);
tab->icon = icon;
tab->text = label;
m_tabs.append(tab);
if (!tt.isEmpty()) {
tab->setToolTip(tt);
} else if (!m_showText) {
tab->setToolTip(label);
}
qobject_cast<QBoxLayout*>(layout())->insertWidget(layout()->count()-(Side==m_pos ? 1 : 0), tab);
}
void FancyTabBar::addSpacer(int size)
{
qobject_cast<QVBoxLayout*>(layout())->insertSpacerItem(layout()->count()-1,
new QSpacerItem(0, size, QSizePolicy::Fixed, QSizePolicy::Maximum));
}
void FancyTabBar::paintTab(QPainter *painter, int tabIndex, bool gtkStyle) const
{
if (!validIndex(tabIndex)) {
qWarning("invalid index");
return;
}
painter->save();
QRect rect = tabRect(tabIndex);
bool selected = (tabIndex == m_currentIndex);
QStyleOptionViewItemV4 styleOpt;
styleOpt.initFrom(this);
styleOpt.state&=~(QStyle::State_Selected|QStyle::State_MouseOver);
styleOpt.state|=QStyle::State_Selected|QStyle::State_Enabled;
styleOpt.rect=rect;
styleOpt.viewItemPosition = QStyleOptionViewItemV4::OnlyOne;
styleOpt.showDecorationSelected=true;
bool drawBgnd=true;
if (!selected && drawBgnd) {
int fader=int(m_tabs[tabIndex]->fader());
if (fader<1) {
drawBgnd=false;
} else {
QColor col(styleOpt.palette.highlight().color());
col.setAlpha(fader);
styleOpt.palette.setColor(styleOpt.palette.currentColorGroup(), QPalette::Highlight, col);
}
}
if (drawBgnd) {
if (!selected && gtkStyle) {
GtkStyle::drawSelection(styleOpt, painter, m_tabs[tabIndex]->fader()/150.0);
} else {
QApplication::style()->drawPrimitive(QStyle::PE_PanelItemViewItem, &styleOpt, painter, 0);
}
}
if (m_showText) {
QString tabText(painter->fontMetrics().elidedText(this->tabText(tabIndex), elideMode(), width()));
QRect tabTextRect(tabRect(tabIndex));
QRect tabIconRect(tabTextRect);
tabIconRect.adjust(+4, +4, -4, -4);
tabTextRect.translate(0, -2);
painter->setPen(selected ? palette().highlightedText().color() : palette().foreground().color());
int textFlags = Qt::AlignCenter | Qt::AlignBottom;
painter->drawText(tabTextRect, textFlags, tabText);
const int textHeight = painter->fontMetrics().height();
tabIconRect.adjust(0, 4, 0, -textHeight);
drawIcon(tabIcon(tabIndex), tabIconRect, painter, QSize(m_iconSize, m_iconSize),
selected && (palette().currentColorGroup()==QPalette::Active ||
(palette().highlightedText().color()==palette().brush(QPalette::Active, QPalette::HighlightedText).color())));
} else {
drawIcon(tabIcon(tabIndex), rect, painter, QSize(m_iconSize, m_iconSize),
selected && (palette().currentColorGroup()==QPalette::Active ||
(palette().highlightedText().color()==palette().brush(QPalette::Active, QPalette::HighlightedText).color())));
}
painter->restore();
}
void FancyTabBar::setCurrentIndex(int index)
{
m_currentIndex = index;
update();
emit currentChanged(m_currentIndex);
}
FancyTabWidget::FancyTabWidget(QWidget* parent, bool allowContext)
: QWidget(parent)
, style_(0)
, tab_bar_(NULL)
, stack_(new QStackedWidget(this))
, side_widget_(new QWidget)
, side_layout_(new QVBoxLayout)
, top_layout_(new QVBoxLayout)
, menu_(0)
, proxy_style_(new FancyTabProxyStyle)
, allowContext_(allowContext)
{
side_layout_->setSpacing(0);
side_layout_->setMargin(0);
side_layout_->addSpacerItem(new QSpacerItem(0, 0, QSizePolicy::Fixed, QSizePolicy::Expanding));
side_widget_->setLayout(side_layout_);
side_widget_->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding);
top_layout_->setMargin(0);
top_layout_->setSpacing(0);
top_layout_->addWidget(stack_);
QHBoxLayout* main_layout = new QHBoxLayout;
main_layout->setMargin(0);
main_layout->setSpacing(1);
main_layout->addWidget(side_widget_);
main_layout->addLayout(top_layout_);
setLayout(main_layout);
setStyle(Side|Large);
}
void FancyTabWidget::AddTab(QWidget* tab, const QIcon& icon, const QString& label, const QString &tt, bool enabled)
{
stack_->addWidget(tab);
items_ << Item(icon, label, tt, enabled);
setMinimumWidth(128);
}
int FancyTabWidget::current_index() const
{
return stack_->currentIndex();
}
QWidget * FancyTabWidget::currentWidget() const
{
return stack_->currentWidget();
}
QWidget * FancyTabWidget::widget(int index) const
{
return stack_->widget(index);
}
int FancyTabWidget::count() const
{
return stack_->count();
}
int FancyTabWidget::visibleCount() const
{
int c=0;
foreach (const Item &i, items_) {
if (i.enabled_) {
c++;
}
}
return c;
}
QSize FancyTabWidget::tabSize() const
{
if (FancyTabBar *bar = qobject_cast<FancyTabBar*>(tab_bar_)) {
return bar->tabSizeHint();
}
return QSize();
}
void FancyTabWidget::SetCurrentIndex(int idx)
{
if (!isEnabled(idx)) {
return;
}
int index=IndexToTab(idx);
if (FancyTabBar* bar = qobject_cast<FancyTabBar*>(tab_bar_)) {
bar->setCurrentIndex(index);
} else if (QTabBar* bar = qobject_cast<QTabBar*>(tab_bar_)) {
bar->setCurrentIndex(index);
ShowWidget(index);
} else {
stack_->setCurrentIndex(idx); // ?? IS this *ever* called???
}
}
void FancyTabWidget::ShowWidget(int index)
{
int idx=TabToIndex(index);
stack_->setCurrentIndex(idx);
emit CurrentChanged(idx);
}
void FancyTabWidget::contextMenuEvent(QContextMenuEvent *e)
{
if (!allowContext_) {
return;
}
// Check we are over tab space...
if (Tab==(style_&Style_Mask)) {
if (QApplication::widgetAt(e->globalPos())!=tab_bar_) {
return;
}
}
else {
switch (style_&Position_Mask) {
case Bot:
if (e->pos().y()<=(side_widget_->pos().y()+(side_widget_->height()-tab_bar_->height()))) {
return;
}
break;
case Top:
if (e->pos().y()>(side_widget_->pos().y()+tab_bar_->height())) {
return;
}
break;
default:
if (Qt::RightToLeft==QApplication::layoutDirection()) {
if (e->pos().x()<=side_widget_->pos().x()) {
return;
}
} else if (e->pos().x()>=side_widget_->rect().right()) {
return;
}
}
}
if (!menu_) {
menu_ = new QMenu(this);
QAction *act=new QAction(i18n("Configure..."), this);
connect(act, SIGNAL(triggered()), SIGNAL(configRequested()));
menu_->addAction(act);
}
menu_->popup(e->globalPos());
}
void FancyTabWidget::setStyle(int s)
{
if(s==style_ && tab_bar_) {
return;
}
// Remove previous tab bar
delete tab_bar_;
tab_bar_ = NULL;
// use_background_ = false;
// Create new tab bar
if (Tab==(s&Style_Mask) || (Small==(s&Style_Mask) && !(s&IconOnly))) {
switch (s&Position_Mask) {
default:
case Side:
MakeTabBar(Qt::RightToLeft==QApplication::layoutDirection() ? QTabBar::RoundedEast : QTabBar::RoundedWest, !(s&IconOnly), true, Small==(s&Style_Mask));
break;
case Top:
MakeTabBar(QTabBar::RoundedNorth, !(s&IconOnly), true, Small==(s&Style_Mask)); break;
case Bot:
MakeTabBar(QTabBar::RoundedSouth, !(s&IconOnly), true, Small==(s&Style_Mask)); break;
}
} else {
FancyTabBar* bar = new FancyTabBar(this, !(s&IconOnly), Small==(s&Style_Mask) ? smallIconSize : largeIconSize,
Side==(s&Position_Mask) ? FancyTabBar::Side : (Top==(s&Position_Mask) ? FancyTabBar::Top : FancyTabBar::Bot));
switch (s&Position_Mask) {
default:
case Side: side_layout_->insertWidget(0, bar); break;
case Bot: top_layout_->insertWidget(1, bar); break;
case Top: top_layout_->insertWidget(0, bar); break;
}
tab_bar_ = bar;
int index=0;
QList<Item>::Iterator it=items_.begin();
QList<Item>::Iterator end=items_.end();
for (; it!=end; ++it) {
if (!(*it).enabled_) {
(*it).index_=-1;
continue;
}
if (Item::Type_Spacer==(*it).type_) {
bar->addSpacer((*it).spacer_size_);
} else {
bar->addTab((*it).tab_icon_, (*it).tab_label_, (*it).tab_tooltip_);
}
(*it).index_=index++;
}
bar->setCurrentIndex(IndexToTab(stack_->currentIndex()));
connect(bar, SIGNAL(currentChanged(int)), SLOT(ShowWidget(int)));
}
tab_bar_->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
style_ = s;
emit styleChanged(s);
update();
}
void FancyTabWidget::ToggleTab(int tab, bool show)
{
if (tab>=0 && tab<items_.count()) {
items_[tab].enabled_=show;
Recreate();
emit TabToggled(tab);
}
}
void FancyTabWidget::MakeTabBar(QTabBar::Shape shape, bool text, bool icons, bool fancy)
{
QTabBar* bar = new QTabBar(this);
bar->setShape(shape);
bar->setDocumentMode(true);
bar->setUsesScrollButtons(true);
if (QTabBar::RoundedWest==shape) {
bar->setIconSize(QSize(smallIconSize, smallIconSize));
}
if (fancy) {
bar->setStyle(proxy_style_.data());
}
if (QTabBar::RoundedNorth==shape) {
top_layout_->insertWidget(0, bar);
} else if (QTabBar::RoundedSouth==shape) {
top_layout_->insertWidget(1, bar);
} else {
side_layout_->insertWidget(0, bar);
}
int index=0;
QList<Item>::Iterator it=items_.begin();
QList<Item>::Iterator end=items_.end();
for (; it!=end; ++it) {
if ((*it).type_ != Item::Type_Tab || !(*it).enabled_) {
(*it).index_=-1;
continue;
}
QString label = (*it).tab_label_;
if (QTabBar::RoundedWest==shape) {
label = QFontMetrics(font()).elidedText(label, elideMode(), 120);
}
int tab_id = -1;
if (icons && text) {
tab_id = bar->addTab((*it).tab_icon_, label);
} else if (icons) {
tab_id = bar->addTab((*it).tab_icon_, QString());
} else if (text) {
tab_id = bar->addTab(label);
}
if (!(*it).tab_tooltip_.isEmpty()) {
bar->setTabToolTip(tab_id, (*it).tab_tooltip_);
} else if (!text) {
bar->setTabToolTip(tab_id, (*it).tab_label_);
}
(*it).index_=index++;
}
bar->setCurrentIndex(IndexToTab(stack_->currentIndex()));
connect(bar, SIGNAL(currentChanged(int)), SLOT(ShowWidget(int)));
tab_bar_ = bar;
}
int FancyTabWidget::TabToIndex(int tab) const
{
for (int i=0; i<items_.count(); ++i) {
if (items_[i].index_==tab) {
return i;
}
}
return 0;
}
void FancyTabWidget::SetIcon(int index, const QIcon &icon)
{
if (index>0 && index<items_.count()) {
items_[index].tab_icon_=icon;
}
}
void FancyTabWidget::SetToolTip(int index, const QString &tt)
{
if (index>0 && index<items_.count()) {
Item &item=items_[index];
item.tab_tooltip_=tt.isEmpty() ? item.tab_label_ : tt;
if (tab_bar_ && -1!=item.index_) {
if (qobject_cast<QTabBar *>(tab_bar_)) {
static_cast<QTabBar *>(tab_bar_)->setTabToolTip(item.index_, item.tab_tooltip_);
} else {
static_cast<FancyTabBar *>(tab_bar_)->setTabToolTip(item.index_, item.tab_tooltip_);
}
}
}
}
void FancyTabWidget::Recreate()
{
int s=style_;
style_=0;
setStyle(s);
}
void FancyTabWidget::setHiddenPages(const QStringList &hidden)
{
QSet<QString> h=hidden.toSet();
if (h==hiddenPages().toSet()) {
return;
}
bool needToRecreate=false;
bool needToSetCurrent=false;
for (int i=0; i<count(); ++i) {
QWidget *w=widget(i);
if (w && items_[i].enabled_==hidden.contains(w->metaObject()->className())) {
items_[i].enabled_=!items_[i].enabled_;
emit TabToggled(i);
needToRecreate=true;
if (i==current_index()) {
needToSetCurrent=true;
}
}
}
if (needToRecreate) {
Recreate();
}
if (needToSetCurrent) {
for (int i=0; i<count(); ++i) {
QWidget *w=widget(i);
if (w && items_[i].enabled_) {
SetCurrentIndex(i);
break;
}
}
}
}
QStringList FancyTabWidget::hiddenPages() const
{
QStringList pages;
for (int i=0; i<count(); ++i) {
if (!isEnabled(i)) {
QWidget *w=widget(i);
if (w) {
pages << w->metaObject()->className();
}
}
}
return pages;
}