Files
cantata/support/pagewidget.cpp
2017-09-10 14:47:26 +01:00

392 lines
13 KiB
C++

/*
* Cantata
*
* Copyright (c) 2011-2017 Craig Drummond <craig.p.drummond@gmail.com>
*
* ----
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; see the file COPYING. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "pagewidget.h"
#include "icon.h"
#include "gtkstyle.h"
#include "dialog.h"
#include <QStyleOption>
#include <QListWidget>
#include <QStackedWidget>
#include <QBoxLayout>
#include <QLabel>
#include <QSizePolicy>
#include <QApplication>
#include <QFontMetrics>
#include <QTextLayout>
#include <QPainter>
#include <QStyledItemDelegate>
static int layoutText(QTextLayout *layout, int maxWidth)
{
qreal height = 0;
int textWidth = 0;
layout->beginLayout();
while (true) {
QTextLine line = layout->createLine();
if (!line.isValid()) {
break;
}
line.setLineWidth(maxWidth);
line.setPosition(QPointF(0, height));
height += line.height();
textWidth = qMax(textWidth, qRound(line.naturalTextWidth() + 0.5));
}
layout->endLayout();
return textWidth;
}
class PageWidgetItemDelegate : public QStyledItemDelegate
{
public:
PageWidgetItemDelegate(QObject *parent, bool std)
: QStyledItemDelegate(parent)
, standard(std)
, underMouse(false)
{
if (!standard) {
int height=QApplication::fontMetrics().height();
iconSize=height>22 ? Icon::stdSize(height*2.5) : 32;
}
}
void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
if (!index.isValid()) {
return;
}
bool mouseOver=option.state&QStyle::State_MouseOver;
bool selected=option.state&QStyle::State_Selected;
if (standard) {
if (GtkStyle::isActive()) {
bool mouseOver=option.state&QStyle::State_MouseOver;
QStyleOptionViewItem opt = option;
initStyleOption(&opt, index);
if (!underMouse) {
mouseOver=false;
}
if (mouseOver) {
opt.showDecorationSelected=true;
GtkStyle::drawSelection(option, painter, selected ? 0.75 : 0.25);
opt.showDecorationSelected=false;
opt.state&=~(QStyle::State_MouseOver|QStyle::State_Selected);
opt.backgroundBrush=QBrush(Qt::transparent);
if (selected) {
opt.palette.setBrush(QPalette::Text, opt.palette.highlightedText());
}
}
QApplication::style()->drawControl(QStyle::CE_ItemViewItem, &opt, painter, opt.widget);
} else {
QStyledItemDelegate::paint(painter, option, index);
}
return;
}
bool gtk=mouseOver && GtkStyle::isActive();
bool drawBgnd=true;
if (!underMouse) {
if (mouseOver && !selected) {
drawBgnd=false;
}
mouseOver=false;
}
const QString text = index.model()->data(index, Qt::DisplayRole).toString();
const QIcon icon = index.model()->data(index, Qt::DecorationRole).value<QIcon>();
const QPixmap pixmap = icon.pixmap(iconSize, iconSize);
QFontMetrics fm = painter->fontMetrics();
QSize layoutSize = pixmap.size() / pixmap.DEVICE_PIXEL_RATIO();
QTextLayout iconTextLayout(text, option.font);
QTextOption textOption(Qt::AlignHCenter);
iconTextLayout.setTextOption(textOption);
int maxWidth = qMax(3 * layoutSize.width(), 8 * fm.height());
layoutText(&iconTextLayout, maxWidth);
QPen pen = painter->pen();
QPalette::ColorGroup cg = option.state & QStyle::State_Enabled
? QPalette::Normal : QPalette::Disabled;
if (cg == QPalette::Normal && !(option.state & QStyle::State_Active)) {
cg = QPalette::Inactive;
}
#ifdef Q_OS_LINUX
if (QPalette::Inactive==cg && option.state & QStyle::State_Enabled && parent() && qobject_cast<QWidget *>(parent())) {
QWidget *pw=static_cast<QWidget *>(parent());
if (pw->topLevelWidget() && pw->topLevelWidget()->isActiveWindow()) {
cg = QPalette::Normal;
}
}
#endif
QStyleOptionViewItem opt(option);
opt.showDecorationSelected = true;
QStyle *style = opt.widget ? opt.widget->style() : QApplication::style();
if (drawBgnd) {
if (mouseOver && gtk) {
GtkStyle::drawSelection(opt, painter, selected ? 0.75 : 0.25);
} else {
style->drawPrimitive(QStyle::PE_PanelItemViewItem, &opt, painter, opt.widget);
}
}
#ifndef Q_OS_WIN
if (selected) {
painter->setPen(option.palette.color(cg, QPalette::HighlightedText));
} else
#endif
{
painter->setPen(option.palette.color(cg, QPalette::Text));
}
painter->drawPixmap(option.rect.x() + (option.rect.width()/2)-(layoutSize.width()/2), option.rect.y() + 5, pixmap);
if (!text.isEmpty()) {
iconTextLayout.draw(painter, QPoint(option.rect.x() + (option.rect.width()/2)-(maxWidth/2), option.rect.y() + layoutSize.height()+7));
}
painter->setPen(pen);
drawFocus(painter, option, option.rect);
}
QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
{
if (standard) {
return QStyledItemDelegate::sizeHint(option, index);
}
if (!index.isValid()) {
return QSize(0, 0);
}
const QString text = index.model()->data(index, Qt::DisplayRole).toString();
const QIcon icon = index.model()->data(index, Qt::DecorationRole).value<QIcon>();
const QPixmap pixmap = icon.pixmap(iconSize, iconSize);
QFontMetrics fm = option.fontMetrics;
int gap = fm.height();
QSize layoutSize = pixmap.size() / pixmap.DEVICE_PIXEL_RATIO();
if (layoutSize.height() == 0) {
/**
* No pixmap loaded yet, we'll use the default icon size in this case.
*/
layoutSize=QSize(iconSize, iconSize);
}
QTextLayout iconTextLayout(text, option.font);
int wt = layoutText(&iconTextLayout, qMax(3 * layoutSize.width(), 8 * fm.height()));
int ht = iconTextLayout.boundingRect().height();
int width, height;
if (text.isEmpty()) {
height = layoutSize.height();
} else {
height = layoutSize.height() + ht + 10;
}
width = qMax(wt, layoutSize.width()) + gap;
return QSize(width, height);
}
void drawFocus(QPainter *painter, const QStyleOptionViewItem &option, const QRect &rect) const
{
if (option.state & QStyle::State_HasFocus) {
QStyleOptionFocusRect o;
o.QStyleOption::operator=(option);
o.rect = rect;
o.state |= QStyle::State_KeyboardFocusChange;
QPalette::ColorGroup cg = (option.state & QStyle::State_Enabled)
? QPalette::Normal : QPalette::Disabled;
o.backgroundColor = option.palette.color(cg, (option.state & QStyle::State_Selected)
? QPalette::Highlight : QPalette::Background);
QApplication::style()->drawPrimitive(QStyle::PE_FrameFocusRect, &o, painter);
}
}
void setUnderMouse(bool u) { underMouse=u; }
bool standardList() const { return standard; }
private:
bool standard;
int iconSize;
bool underMouse;
};
class PageWidgetViewEventHandler : public QObject
{
public:
PageWidgetViewEventHandler(PageWidgetItemDelegate *d, QListWidget *v)
: QObject(v)
, delegate(d)
, view(v)
{
}
protected:
bool eventFilter(QObject *obj, QEvent *event)
{
if (delegate) {
if (QEvent::Enter==event->type()) {
delegate->setUnderMouse(true);
view->viewport()->update();
} else if (QEvent::Leave==event->type()) {
delegate->setUnderMouse(false);
view->viewport()->update();
}
}
return QObject::eventFilter(obj, event);
}
protected:
PageWidgetItemDelegate *delegate;
QListWidget *view;
};
PageWidgetItem::PageWidgetItem(QWidget *p, const QString &header, const QIcon &icon, QWidget *cfg, bool showHeader)
: QWidget(p)
, wid(cfg)
{
QBoxLayout *layout=new QBoxLayout(QBoxLayout::TopToBottom, this);
if (showHeader) {
QBoxLayout *titleLayout=new QBoxLayout(QBoxLayout::LeftToRight, 0);
titleLayout->addWidget(new QLabel("<b>"+header+"</b>", this));
titleLayout->addItem(new QSpacerItem(16, 16, QSizePolicy::Expanding, QSizePolicy::Minimum));
static int iconSize=-1;
if (-1==iconSize) {
iconSize=QApplication::fontMetrics().height();
if (iconSize>20) {
iconSize=Icon::stdSize(iconSize*1.25);
} else {
iconSize=22;
}
}
QLabel *icn=new QLabel(this);
icn->setPixmap(icon.pixmap(iconSize, iconSize));
titleLayout->addWidget(icn);
layout->addLayout(titleLayout);
layout->addItem(new QSpacerItem(8, 8, QSizePolicy::Fixed, QSizePolicy::Fixed));
}
layout->addWidget(cfg);
layout->setMargin(0);
cfg->setParent(this);
adjustSize();
}
PageWidget::PageWidget(QWidget *p, bool listView, bool headers)
: QWidget(p)
, showHeaders(headers)
{
QBoxLayout *layout=new QBoxLayout(QBoxLayout::LeftToRight, this);
list = new QListWidget(p);
stack = new QStackedWidget(p);
connect(list, SIGNAL(currentRowChanged(int)), stack, SLOT(setCurrentIndex(int)));
connect(stack, SIGNAL(currentChanged(int)), this, SIGNAL(currentPageChanged()));
layout->addWidget(list);
layout->addWidget(stack);
layout->setMargin(0);
list->setViewMode(QListView::ListMode);
list->setVerticalScrollMode(QListView::ScrollPerPixel);
list->setMovement(QListView::Static);
PageWidgetItemDelegate *delegate=new PageWidgetItemDelegate(list, listView);
list->setItemDelegate(delegate);
list->setSelectionBehavior(QAbstractItemView::SelectItems);
list->setSelectionMode(QAbstractItemView::SingleSelection);
list->setAttribute(Qt::WA_MouseTracking);
list->installEventFilter(new PageWidgetViewEventHandler(delegate, list));
}
PageWidgetItem * PageWidget::addPage(QWidget *widget, const QString &name, const QIcon &icon, const QString &header)
{
PageWidgetItem *page=new PageWidgetItem(stack, header, icon, widget, showHeaders);
QListWidgetItem *listItem=new QListWidgetItem(name, list);
listItem->setIcon(icon);
stack->addWidget(page);
list->addItem(listItem);
int rows = list->model()->rowCount();
int width = 0;
for (int i = 0; i < rows; ++i) {
QSize rowSize=list->sizeHintForIndex(list->model()->index(i, 0));
width = qMax(width, rowSize.width());
}
width+=static_cast<PageWidgetItemDelegate *>(list->itemDelegate())->standardList() ? 8 : 25;
list->setFixedWidth(width);
QSize stackSize = stack->size();
for (int i = 0; i < stack->count(); ++i) {
const QWidget *widget = stack->widget(i);
if (widget) {
stackSize = stackSize.expandedTo(widget->minimumSizeHint());
}
}
setMinimumHeight(qMax(minimumHeight(), stackSize.height()));
setMinimumWidth(qMax(minimumWidth(), stackSize.width()+width+layout()->spacing()));
list->setCurrentRow(0);
stack->setCurrentIndex(0);
pages.insert(listItem, page);
return page;
}
int PageWidget::count()
{
return list->count();
}
PageWidgetItem * PageWidget::currentPage() const
{
return static_cast<PageWidgetItem *>(stack->currentWidget());
}
void PageWidget::setCurrentPage(PageWidgetItem *item)
{
if (!item) {
return;
}
QMap<QListWidgetItem *, PageWidgetItem*>::ConstIterator it(pages.constBegin());
QMap<QListWidgetItem *, PageWidgetItem*>::ConstIterator end(pages.constEnd());
for (; it!=end; ++it) {
if (it.value()==item) {
list->setCurrentItem(it.key());
}
}
}
void PageWidget::setFocus()
{
list->setFocus();
}