mirror of
https://bitbucket.org/smil3y/kde-workspace.git
synced 2025-02-24 10:52:51 +00:00
338 lines
13 KiB
C++
338 lines
13 KiB
C++
/* This file is part of the KDE libraries and the Kate part.
|
|
*
|
|
* Copyright (C) 2006 Hamish Rodda <rodda@kde.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public License
|
|
* along with this library; see the file COPYING.LIB. If not, write to
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "expandingdelegate.h"
|
|
|
|
#include <QtGui/qtextlayout.h>
|
|
#include <QtGui/QPainter>
|
|
#include <QtGui/QBrush>
|
|
#include <QtGui/qevent.h>
|
|
#include <QTreeView>
|
|
#include <QApplication>
|
|
|
|
#include <kdebug.h>
|
|
|
|
#include "expandingwidgetmodel.h"
|
|
|
|
ExpandingDelegate::ExpandingDelegate(ExpandingWidgetModel* model, QObject* parent)
|
|
: QItemDelegate(parent)
|
|
, m_model(model)
|
|
{
|
|
}
|
|
|
|
//Gets the background-color in the way QItemDelegate does it
|
|
static QColor getUsedBackgroundColor(const QStyleOptionViewItem & option, const QModelIndex& index) {
|
|
if (option.showDecorationSelected && (option.state & QStyle::State_Selected)) {
|
|
QPalette::ColorGroup cg = option.state & QStyle::State_Enabled
|
|
? QPalette::Normal : QPalette::Disabled;
|
|
if (cg == QPalette::Normal && !(option.state & QStyle::State_Active))
|
|
cg = QPalette::Inactive;
|
|
|
|
return option.palette.brush(cg, QPalette::Highlight).color();
|
|
} else {
|
|
QVariant value = index.data(Qt::BackgroundRole);
|
|
if (value.canConvert<QBrush>())
|
|
return qvariant_cast<QBrush>(value).color();
|
|
}
|
|
|
|
return QApplication::palette().base().color();
|
|
}
|
|
|
|
static void dampColors(QColor& col) {
|
|
//Reduce the colors that are less visible to the eye, because they are closer to black when it comes to contrast
|
|
//The most significant color to the eye is green. Then comes red, and then blue, with blue _much_ less significant.
|
|
|
|
col.setBlue(0);
|
|
col.setRed(col.red() / 2);
|
|
}
|
|
|
|
//A hack to compute more eye-focused contrast values
|
|
static double readabilityContrast(QColor foreground, QColor background) {
|
|
dampColors(foreground);
|
|
dampColors(background);
|
|
return abs(foreground.green()-background.green()) + abs(foreground.red()-background.red()) + abs(foreground.blue() - background.blue());
|
|
}
|
|
|
|
void ExpandingDelegate::paint( QPainter * painter, const QStyleOptionViewItem & optionOld, const QModelIndex & index ) const
|
|
{
|
|
QStyleOptionViewItem option(optionOld);
|
|
|
|
m_currentIndex = index;
|
|
|
|
adjustStyle(index, option);
|
|
|
|
if( index.column() == 0 )
|
|
model()->placeExpandingWidget(index);
|
|
|
|
//Make sure the decorations are painted at the top, because the center of expanded items will be filled with the embedded widget.
|
|
if( model()->isPartiallyExpanded(index) == ExpandingWidgetModel::ExpandUpwards )
|
|
m_cachedAlignment = Qt::AlignBottom;
|
|
else
|
|
m_cachedAlignment = Qt::AlignTop;
|
|
|
|
option.decorationAlignment = m_cachedAlignment;
|
|
option.displayAlignment = m_cachedAlignment;
|
|
|
|
//kDebug( 13035 ) << "Painting row " << index.row() << ", column " << index.column() << ", internal " << index.internalPointer() << ", drawselected " << option.showDecorationSelected << ", selected " << (option.state & QStyle::State_Selected);
|
|
|
|
m_cachedHighlights.clear();
|
|
m_backgroundColor = getUsedBackgroundColor(option, index);
|
|
|
|
if (model()->indexIsItem(index) ) {
|
|
m_currentColumnStart = 0;
|
|
m_cachedHighlights = createHighlighting(index, option);
|
|
}
|
|
|
|
/*kDebug( 13035 ) << "Highlights for line:";
|
|
foreach (const QTextLayout::FormatRange& fr, m_cachedHighlights)
|
|
kDebug( 13035 ) << fr.start << " len " << fr.length << " format ";*/
|
|
|
|
QItemDelegate::paint(painter, option, index);
|
|
|
|
///This is a bug workaround for the Qt raster paint engine: It paints over widgets embedded into the viewport when updating due to mouse events
|
|
///@todo report to Qt Software
|
|
if( model()->isExpanded(index) && model()->expandingWidget( index ) )
|
|
model()->expandingWidget( index )->update();
|
|
}
|
|
|
|
QList<QTextLayout::FormatRange> ExpandingDelegate::createHighlighting(const QModelIndex& index, QStyleOptionViewItem& option) const {
|
|
Q_UNUSED( index );
|
|
Q_UNUSED( option );
|
|
return QList<QTextLayout::FormatRange>();
|
|
}
|
|
|
|
QSize ExpandingDelegate::basicSizeHint( const QModelIndex& index ) const {
|
|
return QItemDelegate::sizeHint( QStyleOptionViewItem(), index );
|
|
}
|
|
|
|
QSize ExpandingDelegate::sizeHint ( const QStyleOptionViewItem & option, const QModelIndex & index ) const
|
|
{
|
|
QSize s = QItemDelegate::sizeHint( option, index );
|
|
if( model()->isExpanded(index) && model()->expandingWidget( index ) )
|
|
{
|
|
QWidget* widget = model()->expandingWidget( index );
|
|
QSize widgetSize = widget->size();
|
|
|
|
s.setHeight( widgetSize.height() + s.height() + 10 ); //10 is the sum that must match exactly the offsets used in ExpandingWidgetModel::placeExpandingWidgets
|
|
} else if( model()->isPartiallyExpanded( index ) ) {
|
|
s.setHeight( s.height() + 30 + 10 );
|
|
}
|
|
return s;
|
|
}
|
|
|
|
void ExpandingDelegate::adjustStyle( const QModelIndex& index, QStyleOptionViewItem & option ) const
|
|
{
|
|
Q_UNUSED(index)
|
|
Q_UNUSED(option)
|
|
}
|
|
|
|
void ExpandingDelegate::adjustRect(QRect& rect) const {
|
|
if (!model()->indexIsItem(m_currentIndex) /*&& m_currentIndex.column() == 0*/) {
|
|
|
|
rect.setLeft(model()->treeView()->columnViewportPosition(0));
|
|
int columnCount = model()->columnCount(m_currentIndex.parent());
|
|
|
|
if(!columnCount)
|
|
return;
|
|
rect.setRight(model()->treeView()->columnViewportPosition(columnCount-1) + model()->treeView()->columnWidth(columnCount-1));
|
|
}
|
|
}
|
|
|
|
void ExpandingDelegate::drawDisplay( QPainter * painter, const QStyleOptionViewItem & option, const QRect & _rect, const QString & text ) const
|
|
{
|
|
QRect rect(_rect);
|
|
|
|
adjustRect(rect);
|
|
|
|
QTextLayout layout(text, option.font, painter->device());
|
|
|
|
QList<QTextLayout::FormatRange> additionalFormats;
|
|
|
|
int missingFormats = text.length();
|
|
|
|
for (int i = 0; i < m_cachedHighlights.count(); ++i) {
|
|
if (m_cachedHighlights[i].start + m_cachedHighlights[i].length <= m_currentColumnStart)
|
|
continue;
|
|
|
|
if (!additionalFormats.count())
|
|
if (i != 0 && m_cachedHighlights[i - 1].start + m_cachedHighlights[i - 1].length > m_currentColumnStart) {
|
|
QTextLayout::FormatRange before;
|
|
before.start = 0;
|
|
before.length = m_cachedHighlights[i - 1].start + m_cachedHighlights[i - 1].length - m_currentColumnStart;
|
|
before.format = m_cachedHighlights[i - 1].format;
|
|
additionalFormats.append(before);
|
|
}
|
|
|
|
|
|
QTextLayout::FormatRange format;
|
|
format.start = m_cachedHighlights[i].start - m_currentColumnStart;
|
|
format.length = m_cachedHighlights[i].length;
|
|
format.format = m_cachedHighlights[i].format;
|
|
|
|
additionalFormats.append(format);
|
|
}
|
|
if(!additionalFormats.isEmpty())
|
|
missingFormats = text.length() - (additionalFormats.back().length + additionalFormats.back().start);
|
|
|
|
if (missingFormats > 0) {
|
|
QTextLayout::FormatRange format;
|
|
format.start = text.length() - missingFormats;
|
|
format.length = missingFormats;
|
|
QTextCharFormat fm;
|
|
fm.setForeground(option.palette.text());
|
|
format.format = fm;
|
|
additionalFormats.append(format);
|
|
}
|
|
|
|
if(m_backgroundColor.isValid()) {
|
|
QColor background = m_backgroundColor;
|
|
// kDebug() << text << "background:" << background.name();
|
|
//Now go through the formats, and make sure the contrast background/foreground is readable
|
|
for(int a = 0; a < additionalFormats.size(); ++a) {
|
|
QColor currentBackground = background;
|
|
if(additionalFormats[a].format.hasProperty(QTextFormat::BackgroundBrush))
|
|
currentBackground = additionalFormats[a].format.background().color();
|
|
|
|
QColor currentColor = additionalFormats[a].format.foreground().color();
|
|
|
|
double currentContrast = readabilityContrast(currentColor, currentBackground);
|
|
QColor invertedColor(0xffffffff-additionalFormats[a].format.foreground().color().rgb());
|
|
double invertedContrast = readabilityContrast(invertedColor, currentBackground);
|
|
|
|
// kDebug() << "values:" << invertedContrast << currentContrast << invertedColor.name() << currentColor.name();
|
|
|
|
if(invertedContrast > currentContrast) {
|
|
// kDebug() << text << additionalFormats[a].length << "switching from" << currentColor.name() << "to" << invertedColor.name();
|
|
QBrush b(additionalFormats[a].format.foreground());
|
|
b.setColor(invertedColor);
|
|
additionalFormats[a].format.setForeground(b);
|
|
}
|
|
}
|
|
}
|
|
|
|
for(int a = additionalFormats.size()-1; a >= 0; --a) {
|
|
if(additionalFormats[a].length == 0){
|
|
additionalFormats.removeAt(a);
|
|
}else{
|
|
///For some reason the text-formats seem to be invalid in some way, sometimes
|
|
///@todo Fix this properly, it sucks not copying everything over
|
|
QTextCharFormat fm;
|
|
fm.setForeground(QBrush(additionalFormats[a].format.foreground().color()));
|
|
fm.setBackground(additionalFormats[a].format.background());
|
|
fm.setUnderlineStyle( additionalFormats[a].format.underlineStyle() );
|
|
fm.setUnderlineColor( additionalFormats[a].format.underlineColor() );
|
|
fm.setFontWeight( additionalFormats[a].format.fontWeight() );
|
|
additionalFormats[a].format = fm;
|
|
}
|
|
}
|
|
|
|
// kDebug( 13035 ) << "Highlights for text [" << text << "] col start " << m_currentColumnStart << ":";
|
|
// foreach (const QTextLayout::FormatRange& fr, additionalFormats)
|
|
// kDebug( 13035 ) << fr.start << " len " << fr.length << "foreground" << fr.format.foreground() << "background" << fr.format.background();
|
|
|
|
layout.setAdditionalFormats(additionalFormats);
|
|
|
|
QTextOption to;
|
|
|
|
to.setAlignment( static_cast<Qt::Alignment>(m_cachedAlignment | option.displayAlignment) );
|
|
|
|
to.setWrapMode(QTextOption::WrapAnywhere);
|
|
layout.setTextOption(to);
|
|
|
|
layout.beginLayout();
|
|
QTextLine line = layout.createLine();
|
|
// Leave some extra space when the text is right-aligned
|
|
line.setLineWidth(rect.width() - (option.displayAlignment == Qt::AlignRight ? 8 : 0));
|
|
layout.endLayout();
|
|
|
|
//We need to do some hand layouting here
|
|
if( to.alignment() & Qt::AlignBottom)
|
|
layout.draw(painter, QPoint(rect.left(), rect.bottom() - (int)line.height()) );
|
|
else
|
|
layout.draw(painter, rect.topLeft() );
|
|
|
|
return;
|
|
|
|
//if (painter->fontMetrics().width(text) > textRect.width() && !text.contains(QLatin1Char('\n')))
|
|
//str = elidedText(option.fontMetrics, textRect.width(), option.textElideMode, text);
|
|
//qt_format_text(option.font, textRect, option.displayAlignment, str, 0, 0, 0, 0, painter);
|
|
}
|
|
|
|
void ExpandingDelegate::drawDecoration(QPainter* painter, const QStyleOptionViewItem& option, const QRect& rect, const QPixmap& pixmap) const {
|
|
if (model()->indexIsItem(m_currentIndex) )
|
|
QItemDelegate::drawDecoration(painter, option, rect, pixmap);
|
|
}
|
|
|
|
void ExpandingDelegate::drawBackground ( QPainter * painter, const QStyleOptionViewItem & option, const QModelIndex & index ) const {
|
|
Q_UNUSED(index)
|
|
QStyleOptionViewItem opt = option;
|
|
//initStyleOption(&opt, index);
|
|
//Problem: This isn't called at all, because drawBrackground is not virtual :-/
|
|
QStyle *style = model()->treeView()->style() ? model()->treeView()->style() : QApplication::style();
|
|
style->drawControl(QStyle::CE_ItemViewItem, &opt, painter);
|
|
}
|
|
|
|
ExpandingWidgetModel* ExpandingDelegate::model() const {
|
|
return m_model;
|
|
}
|
|
|
|
void ExpandingDelegate::heightChanged() const {
|
|
}
|
|
|
|
bool ExpandingDelegate::editorEvent ( QEvent * event, QAbstractItemModel * /*model*/, const QStyleOptionViewItem & /*option*/, const QModelIndex & index )
|
|
{
|
|
if( event->type() == QEvent::MouseButtonRelease )
|
|
{
|
|
event->accept();
|
|
model()->setExpanded(index, !model()->isExpanded( index ));
|
|
heightChanged();
|
|
|
|
return true;
|
|
} else {
|
|
event->ignore();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
QList<QTextLayout::FormatRange> ExpandingDelegate::highlightingFromVariantList(const QList<QVariant>& customHighlights) const
|
|
{
|
|
QList<QTextLayout::FormatRange> ret;
|
|
|
|
for (int i = 0; i + 2 < customHighlights.count(); i += 3) {
|
|
if (!customHighlights[i].canConvert(QVariant::Int) || !customHighlights[i+1].canConvert(QVariant::Int) || !customHighlights[i+2].canConvert<QTextFormat>()) {
|
|
kWarning() << "Unable to convert triple to custom formatting.";
|
|
continue;
|
|
}
|
|
|
|
QTextLayout::FormatRange format;
|
|
format.start = customHighlights[i].toInt();
|
|
format.length = customHighlights[i+1].toInt();
|
|
format.format = customHighlights[i+2].value<QTextFormat>().toCharFormat();
|
|
|
|
if(!format.format.isValid())
|
|
kWarning() << "Format is not valid";
|
|
|
|
ret << format;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
#include "moc_expandingdelegate.cpp"
|