X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fqtui%2Fchatline.cpp;h=c1c98da62babd4f9711fd4846d4d8d7dfab82bc4;hp=df218a51ae9d9e73a1e9708a54bcaa29a0cdbb6f;hb=66c5feaf7c9f480f06dfb45df2cf54e44c8b487b;hpb=176d22d49934223b9279719ac5d9e7c03e530d40 diff --git a/src/qtui/chatline.cpp b/src/qtui/chatline.cpp index df218a51..c1c98da6 100644 --- a/src/qtui/chatline.cpp +++ b/src/qtui/chatline.cpp @@ -1,5 +1,5 @@ /*************************************************************************** - * Copyright (C) 2005-08 by the Quassel Project * + * Copyright (C) 2005-09 by the Quassel Project * * devel@quassel-irc.org * * * * This program is free software; you can redistribute it and/or modify * @@ -34,22 +34,6 @@ #include "qtuisettings.h" #include "qtuistyle.h" -// ChatLine::ChatLine(int row, QAbstractItemModel *model, QGraphicsItem *parent) -// : QGraphicsItem(parent), -// _row(row), // needs to be set before the items -// _model(model), -// _contentsItem(this), -// _senderItem(this), -// _timestampItem(this), -// _width(0), -// _height(0), -// _selection(0) -// { -// Q_ASSERT(model); -// QModelIndex index = model->index(row, ChatLineModel::ContentsColumn); -// setHighlighted(model->data(index, MessageModel::FlagsRole).toInt() & Message::Highlight); -// } - ChatLine::ChatLine(int row, QAbstractItemModel *model, const qreal &width, const qreal ×tampWidth, const qreal &senderWidth, const qreal &contentsWidth, @@ -67,14 +51,10 @@ ChatLine::ChatLine(int row, QAbstractItemModel *model, { Q_ASSERT(model); QModelIndex index = model->index(row, ChatLineModel::ContentsColumn); + setZValue(0); setHighlighted(model->data(index, MessageModel::FlagsRole).toInt() & Message::Highlight); } -QRectF ChatLine::boundingRect () const { - //return childrenBoundingRect(); - return QRectF(0, 0, _width, _height); -} - ChatItem &ChatLine::item(ChatLineModel::ColumnType column) { switch(column) { case ChatLineModel::TimestampColumn: @@ -88,60 +68,64 @@ ChatItem &ChatLine::item(ChatLineModel::ColumnType column) { } } -// WARNING: setColumns should not be used without either: -// a) calling prepareGeometryChange() immediately before setColumns() -// b) calling Chatline::setPos() immediately afterwards -// -// NOTE: senderPos and contentsPos are in ChatLines coordinate system! -qreal ChatLine::setColumns(const qreal ×tampWidth, const qreal &senderWidth, const qreal &contentsWidth, - const QPointF &senderPos, const QPointF &contentsPos) { - _height = _contentsItem.setGeometryByWidth(contentsWidth); - _senderItem.setGeometry(senderWidth, _height); +// NOTE: senderPos is in ChatLines coordinate system! +void ChatLine::setFirstColumn(const qreal ×tampWidth, const qreal &senderWidth, const QPointF &senderPos) { + _timestampItem.prepareGeometryChange(); _timestampItem.setGeometry(timestampWidth, _height); - + // senderItem doesn't need a geom change as it's Pos is changed (ensured by void ChatScene::firstHandlePositionChanged(qreal xpos)) + _senderItem.setGeometry(senderWidth, _height); _senderItem.setPos(senderPos); +} + +// NOTE: contentsPos is in ChatLines coordinate system! +void ChatLine::setSecondColumn(const qreal &senderWidth, const qreal &contentsWidth, + const QPointF &contentsPos, qreal &linePos) { + // contentsItem doesn't need a geom change as it's Pos is changed (ensured by void ChatScene::firstHandlePositionChanged(qreal xpos)) + qreal height = _contentsItem.setGeometryByWidth(contentsWidth); + linePos -= height; + bool needGeometryChange = linePos == pos().y(); + + if(needGeometryChange) { + _timestampItem.prepareGeometryChange(); + _senderItem.prepareGeometryChange(); + } + _timestampItem.setHeight(height); + _senderItem.setGeometry(senderWidth, height); + _contentsItem.setPos(contentsPos); - _contentsItem.clearLayout(); - _senderItem.clearLayout(); - _timestampItem.clearLayout(); + if(needGeometryChange) + prepareGeometryChange(); - return _height; -} + _height = height; -// WARNING: setGeometryByWidth should not be used without either: -// a) calling prepareGeometryChange() immediately before setColumns() -// b) calling Chatline::setPos() immediately afterwards -qreal ChatLine::setGeometryByWidth(const qreal &width, const qreal &contentsWidth) { - _width = width; - _height = _contentsItem.setGeometryByWidth(contentsWidth); - _timestampItem.setHeight(_height); - _senderItem.setHeight(_height); - _contentsItem.clearLayout(); - return _height; + setPos(0, linePos); } -qreal ChatLine::setGeometryByWidth(qreal width) { - if(width != _width) - prepareGeometryChange(); - - ColumnHandleItem *firstColumnHandle = chatScene()->firstColumnHandle(); - ColumnHandleItem *secondColumnHandle = chatScene()->secondColumnHandle(); +void ChatLine::setGeometryByWidth(const qreal &width, const qreal &contentsWidth, qreal &linePos) { + qreal height = _contentsItem.setGeometryByWidth(contentsWidth); + linePos -= height; + bool needGeometryChange = (height != _height || width != _width); - _height = _contentsItem.setGeometryByWidth(width - secondColumnHandle->sceneRight()); - _timestampItem.setGeometry(firstColumnHandle->sceneLeft(), _height); - _senderItem.setGeometry(secondColumnHandle->sceneLeft() - firstColumnHandle->sceneRight(), _height); + if(height != _height) { + _timestampItem.prepareGeometryChange(); + _timestampItem.setHeight(height); + _senderItem.prepareGeometryChange(); + _senderItem.setHeight(height); + } - _senderItem.setPos(firstColumnHandle->sceneRight(), 0); - _contentsItem.setPos(secondColumnHandle->sceneRight(), 0); + if(needGeometryChange) { + prepareGeometryChange(); + _height = height; + _width = width; + } - _width = width; - return _height; + setPos(0, linePos); // set pos is _very_ cheap if nothing changes. } void ChatLine::setSelected(bool selected, ChatLineModel::ColumnType minColumn) { if(selected) { - quint8 sel = (_selection & 0x80) | 0x40 | minColumn; + quint8 sel = (_selection & Highlighted) | Selected | minColumn; if(sel != _selection) { _selection = sel; for(int i = 0; i < minColumn; i++) @@ -151,7 +135,7 @@ void ChatLine::setSelected(bool selected, ChatLineModel::ColumnType minColumn) { update(); } } else { - quint8 sel = _selection & 0x80; + quint8 sel = _selection & Highlighted; if(sel != _selection) { _selection = sel; for(int i = 0; i <= ChatLineModel::ContentsColumn; i++) @@ -162,38 +146,50 @@ void ChatLine::setSelected(bool selected, ChatLineModel::ColumnType minColumn) { } void ChatLine::setHighlighted(bool highlighted) { - if(highlighted) _selection |= 0x80; - else _selection &= 0x7f; + if(highlighted) _selection |= Highlighted; + else _selection &= ~Highlighted; update(); } void ChatLine::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) { Q_UNUSED(option); Q_UNUSED(widget); - if(_selection & Highlighted) { - painter->fillRect(boundingRect(), QBrush(QtUi::style()->highlightColor())); + + const QAbstractItemModel *model_ = model(); + QModelIndex myIdx = model_->index(row(), 0); + Message::Type type = (Message::Type)myIdx.data(MessageModel::TypeRole).toInt(); + UiStyle::MessageLabel label = (UiStyle::MessageLabel)myIdx.data(ChatLineModel::MsgLabelRole).toInt(); + + QTextCharFormat msgFmt = QtUi::style()->format(UiStyle::formatType(type), label); + if(msgFmt.hasProperty(QTextFormat::BackgroundBrush)) { + painter->fillRect(boundingRect(), msgFmt.background()); } + if(_selection & Selected) { - qreal left = item((ChatLineModel::ColumnType)(_selection & 0x3f)).x(); - QRectF selectRect(left, 0, width() - left, height()); - painter->fillRect(selectRect, QApplication::palette().brush(QPalette::Highlight)); + QTextCharFormat selFmt = QtUi::style()->format(UiStyle::formatType(type), label | UiStyle::Selected); + if(selFmt.hasProperty(QTextFormat::BackgroundBrush)) { + qreal left = item((ChatLineModel::ColumnType)(_selection & ItemMask)).x(); + QRectF selectRect(left, 0, width() - left, height()); + painter->fillRect(selectRect, selFmt.background()); + } } // new line marker - const QAbstractItemModel *model_ = model(); - if(model_ && row() > 0) { + if(model_ && row() > 0 && chatScene()->isSingleBufferScene()) { QModelIndex prevRowIdx = model_->index(row() - 1, 0); - MsgId msgId = model_->data(prevRowIdx, MessageModel::MsgIdRole).value(); - Message::Flags flags = (Message::Flags)model_->data(model_->index(row(), 0), MessageModel::FlagsRole).toInt(); + MsgId prevMsgId = prevRowIdx.data(MessageModel::MsgIdRole).value(); + MsgId myMsgId = myIdx.data(MessageModel::MsgIdRole).value(); + Message::Flags flags = (Message::Flags)myIdx.data(MessageModel::FlagsRole).toInt(); + // don't show the marker if we wrote that new line if(!(flags & Message::Self)) { - BufferId bufferId = model_->data(prevRowIdx, MessageModel::BufferIdRole).value(); - if(msgId == Client::networkModel()->lastSeenMsgId(bufferId) && chatScene()->isSingleBufferScene()) { - QtUiStyleSettings s("Colors"); - QLinearGradient gradient(0, 0, 0, height()); - gradient.setColorAt(0, s.value("newMsgMarkerFG", QColor(Qt::red)).value()); - gradient.setColorAt(0.1, Qt::transparent); - painter->fillRect(boundingRect(), gradient); + BufferId bufferId = BufferId(chatScene()->idString().toInt()); + MsgId lastSeenMsgId = Client::networkModel()->lastSeenMarkerMsgId(bufferId); + if(lastSeenMsgId < myMsgId && lastSeenMsgId >= prevMsgId) { + QLinearGradient gradient(0, 0, 0, contentsItem().fontMetrics()->lineSpacing()); + gradient.setColorAt(0, QtUi::style()->brush(UiStyle::MarkerLine).color()); // FIXME: Use full (gradient?) brush instead of just the color + gradient.setColorAt(0.1, Qt::transparent); + painter->fillRect(boundingRect(), gradient); } } }