293 lines
12 KiB
C++
293 lines
12 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2009 Martin Gräßlin <kde@martin-graesslin.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. If not, see <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
// own
|
|
#include "clientitemdelegate.h"
|
|
// tabbox
|
|
#include "clientmodel.h"
|
|
#include "itemlayoutconfig.h"
|
|
// Qt
|
|
#include <QPainter>
|
|
// KDE
|
|
#include <kephal/screens.h>
|
|
#include <KGlobalSettings>
|
|
#include <KIconEffect>
|
|
#include <KIconLoader>
|
|
#include <Plasma/FrameSvg>
|
|
#include <Plasma/Theme>
|
|
|
|
namespace KWin
|
|
{
|
|
namespace TabBox
|
|
{
|
|
|
|
ClientItemDelegate::ClientItemDelegate(QObject* parent)
|
|
: QAbstractItemDelegate(parent)
|
|
, m_showSelection(true)
|
|
{
|
|
m_frame = new Plasma::FrameSvg(this);
|
|
m_frame->setImagePath("widgets/viewitem");
|
|
m_frame->setElementPrefix("hover");
|
|
m_frame->setCacheAllRenderedFrames(true);
|
|
m_frame->setEnabledBorders(Plasma::FrameSvg::AllBorders);
|
|
}
|
|
|
|
ClientItemDelegate::~ClientItemDelegate()
|
|
{
|
|
}
|
|
|
|
void ClientItemDelegate::setConfig(const KWin::TabBox::ItemLayoutConfig& config)
|
|
{
|
|
m_config = config;
|
|
}
|
|
|
|
QSize ClientItemDelegate::sizeHint(const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
{
|
|
if (!index.isValid())
|
|
return QSize(0, 0);
|
|
|
|
// check for empty client list
|
|
if (index.model()->data(index, ClientModel::EmptyRole).toBool()) {
|
|
QFont font = KGlobalSettings::generalFont();
|
|
font.setBold(true);
|
|
font.setPointSize(font.pointSize() + 4);
|
|
QFontMetrics fm(font);
|
|
QString text = index.model()->data(index, Qt::DisplayRole).toString();
|
|
qreal width = 20.0 + fm.width(text);
|
|
qreal height = fm.boundingRect(text).height();
|
|
return QSize(width, height);
|
|
}
|
|
|
|
qreal width = 0.0;
|
|
qreal height = 0.0;
|
|
for (int i = 0; i < m_config.count(); i++) {
|
|
QSizeF row = rowSize(index, i);
|
|
width = qMax<qreal>(width, row.width());
|
|
height += row.height();
|
|
}
|
|
qreal left, top, right, bottom;
|
|
m_frame->getMargins(left, top, right, bottom);
|
|
|
|
// find icon elements which have a row span
|
|
for (int i = 0; i < m_config.count(); i++) {
|
|
ItemLayoutConfigRow row = m_config.row(i);
|
|
for (int j = 0; j < row.count(); j++) {
|
|
ItemLayoutConfigRowElement element = row.element(j);
|
|
if (element.type() == ItemLayoutConfigRowElement::ElementIcon && element.isRowSpan())
|
|
height = qMax<qreal>(height, element.iconSize().height());
|
|
}
|
|
}
|
|
if (!option.rect.isValid()) {
|
|
// item may not be bigger than the screen geometry
|
|
const QRect screenRect = Kephal::ScreenUtils::screenGeometry(tabBox->activeScreen());
|
|
width = qMin(width, qreal(screenRect.width()));
|
|
height = qMin(height, qreal(screenRect.height()));
|
|
} else {
|
|
width = qMin(width, qreal(option.rect.width()));
|
|
height = qMin(height, qreal(option.rect.height()));
|
|
}
|
|
return QSize(width + left + right, height + top + bottom);
|
|
}
|
|
|
|
void ClientItemDelegate::paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
{
|
|
if (!index.isValid())
|
|
return;
|
|
|
|
// check for empty client list
|
|
if (index.model()->data(index, ClientModel::EmptyRole).toBool()) {
|
|
painter->save();
|
|
QFont font = KGlobalSettings::generalFont();
|
|
font.setBold(true);
|
|
font.setPointSize(font.pointSize() + 4);
|
|
painter->setFont(font);
|
|
painter->drawText(option.rect, Qt::AlignCenter | Qt::TextSingleLine,
|
|
index.model()->data(index, Qt::DisplayRole).toString());
|
|
painter->restore();
|
|
return;
|
|
}
|
|
|
|
qreal left, top, right, bottom;
|
|
m_frame->getMargins(left, top, right, bottom);
|
|
|
|
qreal y = option.rect.top() + top;
|
|
for (int i = 0; i < m_config.count(); i++) {
|
|
qreal rowHeight = rowSize(index, i).height();
|
|
qreal x = option.rect.left() + left ;
|
|
ItemLayoutConfigRow row = m_config.row(i);
|
|
for (int j = 0; j < row.count(); j++) {
|
|
ItemLayoutConfigRowElement element = row.element(j);
|
|
switch(element.type()) {
|
|
case ItemLayoutConfigRowElement::ElementClientName:
|
|
x += paintTextElement(painter, option, index, element, x, y, rowHeight,
|
|
index.model()->data(index, ClientModel::CaptionRole).toString());
|
|
break;
|
|
case ItemLayoutConfigRowElement::ElementDesktopName: {
|
|
if (tabBox->config().tabBoxMode() == TabBoxConfig::ClientTabBox &&
|
|
((tabBox->config().clientListMode() == TabBoxConfig::CurrentDesktopClientList) ||
|
|
(tabBox->config().clientListMode() == TabBoxConfig::CurrentDesktopApplicationList)))
|
|
break;
|
|
x += paintTextElement(painter, option, index, element, x, y, rowHeight,
|
|
index.model()->data(index, ClientModel::DesktopNameRole).toString());
|
|
break;
|
|
}
|
|
case ItemLayoutConfigRowElement::ElementIcon: {
|
|
TabBoxClient* client = static_cast< TabBoxClient* >(index.model()->data(
|
|
index, ClientModel::ClientRole).value<void *>());
|
|
qreal rectWidth = (qreal)option.rect.width();
|
|
qreal maxWidth = qMin<qreal>(element.width(), rectWidth);
|
|
if (element.isStretch())
|
|
maxWidth = qMax<qreal>(maxWidth, option.rect.left() + option.rect.width() - x);
|
|
qreal iconX = x + maxWidth * 0.5 - element.iconSize().width() * 0.5;
|
|
qreal iconY = y + rowHeight * 0.5 - element.iconSize().height() * 0.5;
|
|
if (element.isRowSpan())
|
|
iconY = option.rect.top() + option.rect.height() * 0.5 - element.iconSize().height() * 0.5;
|
|
QRectF iconRect = QRectF(iconX, iconY, element.iconSize().width(), element.iconSize().height());
|
|
QPixmap icon = client->icon(element.iconSize().toSize());
|
|
if (!icon.isNull()) {
|
|
if (m_config.isHighlightSelectedIcons() && option.state & QStyle::State_Selected) {
|
|
KIconEffect *effect = KIconLoader::global()->iconEffect();
|
|
icon = effect->apply(icon, KIconLoader::Desktop, KIconLoader::ActiveState);
|
|
}
|
|
if (m_config.isGrayscaleDeselectedIcons() && !(option.state & QStyle::State_Selected)) {
|
|
KIconEffect *effect = KIconLoader::global()->iconEffect();
|
|
icon = effect->apply(icon, KIconLoader::Desktop, KIconLoader::DisabledState);
|
|
}
|
|
QRectF sourceRect = QRectF(QPointF(0.0, 0.0), element.iconSize());
|
|
if (icon.width() > element.iconSize().width()) {
|
|
// if icon is bigger than our region, scale it down
|
|
sourceRect = QRectF(0.0, 0.0, icon.width(), icon.height());
|
|
} else if (icon.width() < element.iconSize().width()) {
|
|
// don't scale - center in requested area
|
|
sourceRect = QRectF(0.0, 0.0, icon.width(), icon.height());
|
|
iconRect = QRectF(iconX + (element.iconSize().width() - icon.width()) / 2,
|
|
iconY + (element.iconSize().height() - icon.height()) / 2,
|
|
icon.width(), icon.height());
|
|
}
|
|
painter->drawPixmap(iconRect, icon, sourceRect);
|
|
}
|
|
x += element.width();
|
|
break;
|
|
}
|
|
case ItemLayoutConfigRowElement::ElementEmpty:
|
|
x += element.width();
|
|
break;
|
|
default:
|
|
break; // do nothing
|
|
}
|
|
}
|
|
y += rowHeight;
|
|
}
|
|
}
|
|
|
|
qreal ClientItemDelegate::paintTextElement(QPainter* painter, const QStyleOptionViewItem& option,
|
|
const QModelIndex& index, const ItemLayoutConfigRowElement& element,
|
|
const qreal& x, const qreal& y, const qreal& rowHeight, QString text) const
|
|
{
|
|
painter->save();
|
|
QFont font = KGlobalSettings::generalFont();
|
|
if (element.isSmallTextSize())
|
|
font = KGlobalSettings::smallestReadableFont();
|
|
font.setBold(element.isBold());
|
|
font.setItalic(element.isItalic());
|
|
text = element.prefix() + text + element.suffix();
|
|
if (index.model()->data(index, ClientModel::MinimizedRole).toBool()) {
|
|
text = element.prefixMinimized() + text + element.suffixMinimized();
|
|
if (element.isItalicMinimized())
|
|
font.setItalic(true);
|
|
}
|
|
painter->setFont(font);
|
|
painter->setPen(Plasma::Theme::defaultTheme()->color(Plasma::Theme::TextColor));
|
|
qreal width = element.width();
|
|
if (element.isStretch()) {
|
|
qreal left, top, right, bottom;
|
|
m_frame->getMargins(left, top, right, bottom);
|
|
width = option.rect.left() + option.rect.width() - x - right;
|
|
}
|
|
text = QFontMetricsF(font).elidedText(text, Qt::ElideMiddle, width);
|
|
QRectF rect = QRectF(x, y, width, rowHeight);
|
|
painter->drawText(rect, element.alignment() | Qt::TextSingleLine, text);
|
|
painter->restore();
|
|
return width;
|
|
}
|
|
|
|
QSizeF ClientItemDelegate::rowSize(const QModelIndex& index, int row) const
|
|
{
|
|
ItemLayoutConfigRow currentRow = m_config.row(row);
|
|
qreal rowWidth = 0.0;
|
|
qreal rowHeight = 0.0;
|
|
for (int j = 0; j < currentRow.count(); j++) {
|
|
ItemLayoutConfigRowElement element = currentRow.element(j);
|
|
switch(element.type()) {
|
|
case ItemLayoutConfigRowElement::ElementClientName: {
|
|
QSizeF size = textElementSizeHint(index, element,
|
|
index.model()->data(index, ClientModel::CaptionRole).toString());
|
|
rowWidth += size.width();
|
|
rowHeight = qMax<qreal>(rowHeight, size.height());
|
|
break;
|
|
}
|
|
case ItemLayoutConfigRowElement::ElementDesktopName: {
|
|
QSizeF size = textElementSizeHint(index, element,
|
|
index.model()->data(index, ClientModel::DesktopNameRole).toString());
|
|
if (tabBox->config().tabBoxMode() == TabBoxConfig::ClientTabBox &&
|
|
((tabBox->config().clientListMode() == TabBoxConfig::CurrentDesktopClientList) ||
|
|
(tabBox->config().clientListMode() == TabBoxConfig::CurrentDesktopApplicationList)))
|
|
size = QSizeF(0, 0);
|
|
rowWidth += size.width();
|
|
rowHeight = qMax<qreal>(rowHeight, size.height());
|
|
break;
|
|
}
|
|
case ItemLayoutConfigRowElement::ElementIcon:
|
|
rowWidth += qMax<qreal>(element.iconSize().width(), element.width());
|
|
if (!element.isRowSpan())
|
|
rowHeight = qMax<qreal>(rowHeight, element.iconSize().height());
|
|
break;
|
|
case ItemLayoutConfigRowElement::ElementEmpty:
|
|
rowWidth += element.width();
|
|
break;
|
|
default:
|
|
break; // do nothing
|
|
}
|
|
}
|
|
return QSizeF(rowWidth, rowHeight);
|
|
}
|
|
|
|
QSizeF ClientItemDelegate::textElementSizeHint(const QModelIndex& index, const ItemLayoutConfigRowElement& element, QString text) const
|
|
{
|
|
QFont font = KGlobalSettings::generalFont();
|
|
if (element.isSmallTextSize())
|
|
font = KGlobalSettings::smallestReadableFont();
|
|
font.setBold(element.isBold());
|
|
font.setItalic(element.isItalic());
|
|
text = element.prefix() + text + element.suffix();
|
|
if (index.model()->data(index, ClientModel::MinimizedRole).toBool()) {
|
|
text = element.prefixMinimized() + text + element.suffixMinimized();
|
|
if (element.isItalicMinimized())
|
|
font.setItalic(true);
|
|
}
|
|
QFontMetricsF fm(font);
|
|
qreal width = element.width();
|
|
if (element.isStretch())
|
|
width = fm.width(text);
|
|
qreal height = fm.boundingRect(text).height();
|
|
return QSizeF(width, height);
|
|
}
|
|
|
|
} // namespace Tabbox
|
|
} // namespace KWin
|