1 /***************************************************************************
2 * Copyright (C) 2005-08 by the Quassel Project *
3 * devel@quassel-irc.org *
5 * This program is free software; you can redistribute it and/or modify *
6 * it under the terms of the GNU General Public License as published by *
7 * the Free Software Foundation; either version 2 of the License, or *
8 * (at your option) version 3. *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the *
17 * Free Software Foundation, Inc., *
18 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
19 ***************************************************************************/
21 #include <QFontMetrics>
22 #include <QGraphicsSceneMouseEvent>
24 #include <QTextLayout>
27 #include "chatlinemodel.h"
30 ChatItem::ChatItem(const QPersistentModelIndex &index_, QGraphicsItem *parent) : QGraphicsItem(parent), _index(index_) {
31 QFontMetricsF *metrics = QtUi::style()->fontMetrics(data(ChatLineModel::FormatRole).value<UiStyle::FormatList>().at(0).second);
32 _lineHeight = metrics->lineSpacing();
33 _lineLeading = metrics->leading();
37 ChatItem::~ChatItem() {
41 QVariant ChatItem::data(int role) const {
42 if(!_index.isValid()) {
43 qWarning() << "ChatItem::data(): Model index is invalid!" << _index;
46 return _index.data(role);
49 int ChatItem::setWidth(int w) {
51 if(w == _boundingRect.width()) return _boundingRect.height();
52 int h = heightForWidth(w);
53 _boundingRect.setWidth(w);
54 _boundingRect.setHeight(h);
55 if(haveLayout()) updateLayout();
59 int ChatItem::heightForWidth(int width) {
60 if(data(ChatLineModel::ColumnTypeRole).toUInt() != ChatLineModel::ContentsColumn)
61 return _lineHeight; // only contents can be multi-line
63 ChatLineModel::WrapList wrapList = data(ChatLineModel::WrapListRole).value<ChatLineModel::WrapList>();
66 for(int i = 0; i < wrapList.count(); i++) {
67 w += wrapList.at(i).width;
69 w += wrapList.at(i).trailing;
73 w = wrapList.at(i).width;
79 return lines * _lineHeight;
82 void ChatItem::layout() {
83 if(haveLayout()) return;
84 _layout = new QTextLayout(data(MessageModel::DisplayRole).toString());
86 // Convert format information into a FormatRange
87 QList<QTextLayout::FormatRange> formatRanges;
88 UiStyle::FormatList formatList = data(MessageModel::FormatRole).value<UiStyle::FormatList>();
89 QTextLayout::FormatRange range;
91 for(i = 0; i < formatList.count(); i++) {
92 range.format = QtUi::style()->mergedFormat(formatList.at(i).second);
93 range.start = formatList.at(i).first;
94 if(i > 0) formatRanges.last().length = range.start - formatRanges.last().start;
95 formatRanges.append(range);
97 if(i > 0) formatRanges.last().length = _layout->text().length() - formatRanges.last().start;
98 _layout->setAdditionalFormats(formatRanges);
102 void ChatItem::updateLayout() {
103 if(!haveLayout()) layout();
107 _layout->beginLayout();
109 QTextLine line = _layout->createLine();
113 line.setLineWidth(width());
115 line.setPosition(QPointF(0, h));
118 _layout->endLayout();
121 void ChatItem::clearLayout() {
126 void ChatItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) {
127 Q_UNUSED(option); Q_UNUSED(widget);
129 _layout->draw(painter, QPointF(0,0), QVector<QTextLayout::FormatRange>(), boundingRect());
130 painter->drawRect(boundingRect());
132 QVariantList wrapList = data(ChatLineModel::WrapListRole).toList();
133 for(int i = 2; i < wrapList.count(); i+=2) {
134 QRect r(wrapList[i-1].toUInt(), 0, wrapList[i+1].toUInt() - wrapList[i-1].toUInt(), _lineHeight);
135 painter->drawRect(r);
140 void ChatItem::layout() {
141 if(!_layout.additionalFormats().count()) return; // no text set
142 if(_width <= 0) return;
143 prepareGeometryChange();
144 QFontMetrics metrics(_layout.additionalFormats()[0].format.font());
145 int leading = metrics.leading();
147 _layout.setTextOption(textOption());
148 _layout.beginLayout();
150 QTextLine line = _layout.createLine();
151 if(!line.isValid()) break;
152 line.setLineWidth(_width);
153 if(textOption().wrapMode() != QTextOption::NoWrap && line.naturalTextWidth() > _width) {
154 // word did not fit, we need to wrap it in the middle
155 // this is a workaround for Qt failing to handle WrapAtWordBoundaryOrAnywhere correctly
156 QTextOption::WrapMode mode = textOption().wrapMode();
157 textOption().setWrapMode(QTextOption::WrapAnywhere);
158 _layout.setTextOption(textOption());
159 line.setLineWidth(_width);
160 textOption().setWrapMode(mode);
161 _layout.setTextOption(textOption());
164 line.setPosition(QPoint(0, height));
165 height += line.height();
169 } QDateTime _timestamp;
173 QRectF ChatItem::boundingRect() const {
174 return _layout.boundingRect();
177 void ChatItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) {
178 Q_UNUSED(option); Q_UNUSED(widget);
179 _layout.draw(painter, QPointF(0, 0));
185 void ChatItem::mouseMoveEvent ( QGraphicsSceneMouseEvent * event ) {
186 qDebug() << (void*)this << "moving" << event->pos();
187 if(event->pos().y() < 0) {
188 QTextCursor cursor(document());
189 //cursor.insertText("foo");
190 //cursor.select(QTextCursor::Document);
192 } else QGraphicsTextItem::mouseMoveEvent(event);