+ formats.append(selectFmt);
+ }
+ layout()->draw(painter, QPointF(0,0), formats, boundingRect());
+
+ // Debuging Stuff
+ // uncomment the following lines to draw the bounding rect and the row number in alternating colors
+// if(row() % 2)
+// painter->setPen(Qt::red);
+// else
+// painter->setPen(Qt::blue);
+// QString rowString = QString::number(row());
+// QRect rowRect = painter->fontMetrics().boundingRect(rowString);
+// QPointF topPoint = _boundingRect.topLeft();
+// topPoint.ry() += rowRect.height();
+// painter->drawText(topPoint, rowString);
+// QPointF bottomPoint = _boundingRect.bottomRight();
+// bottomPoint.rx() -= rowRect.width();
+// painter->drawText(bottomPoint, rowString);
+// painter->drawRect(_boundingRect.adjusted(0, 0, -1, -1));
+}
+
+qint16 ChatItem::posToCursor(const QPointF &pos) {
+ if(pos.y() > height()) return data(MessageModel::DisplayRole).toString().length();
+ if(pos.y() < 0) return 0;
+ if(!hasLayout())
+ updateLayout();
+ for(int l = layout()->lineCount() - 1; l >= 0; l--) {
+ QTextLine line = layout()->lineAt(l);
+ if(pos.y() >= line.y()) {
+ return line.xToCursor(pos.x(), QTextLine::CursorOnCharacter);
+ }
+ }
+ return 0;
+}
+
+void ChatItem::setFullSelection() {
+ if(_selectionMode != FullSelection) {
+ _selectionMode = FullSelection;
+ update();
+ }
+}
+
+void ChatItem::clearSelection() {
+ _selectionMode = NoSelection;
+ update();
+}
+
+void ChatItem::continueSelecting(const QPointF &pos) {
+ _selectionMode = PartialSelection;
+ _selectionEnd = posToCursor(pos);
+ update();
+}
+
+QList<QRectF> ChatItem::findWords(const QString &searchWord, Qt::CaseSensitivity caseSensitive) {
+ QList<QRectF> resultList;
+ const QAbstractItemModel *model_ = model();
+ if(!model_)
+ return resultList;
+
+ QString plainText = model_->data(model_->index(row(), column()), MessageModel::DisplayRole).toString();
+ QList<int> indexList;
+ int searchIdx = plainText.indexOf(searchWord, 0, caseSensitive);
+ while(searchIdx != -1) {
+ indexList << searchIdx;
+ searchIdx = plainText.indexOf(searchWord, searchIdx + 1, caseSensitive);
+ }
+
+ bool hadLayout = hasLayout();
+ if(!hadLayout)
+ updateLayout();
+
+ foreach(int idx, indexList) {
+ QTextLine line = layout()->lineForTextPosition(idx);
+ qreal x = line.cursorToX(idx);
+ qreal width = line.cursorToX(idx + searchWord.count()) - x;
+ qreal height = line.height();
+ qreal y = height * line.lineNumber();
+ resultList << QRectF(x, y, width, height);
+ }
+
+ if(!hadLayout)
+ clearLayout();
+ return resultList;
+}
+
+void ChatItem::mousePressEvent(QGraphicsSceneMouseEvent *event) {
+ if(event->buttons() == Qt::LeftButton) {
+ chatScene()->setSelectingItem(this);
+ _selectionStart = _selectionEnd = posToCursor(event->pos());
+ _selectionMode = NoSelection; // will be set to PartialSelection by mouseMoveEvent
+ update();
+ event->accept();
+ } else {
+ event->ignore();
+ }
+}
+
+void ChatItem::mouseMoveEvent(QGraphicsSceneMouseEvent *event) {
+ if(event->buttons() == Qt::LeftButton) {
+ if(contains(event->pos())) {
+ qint16 end = posToCursor(event->pos());
+ if(end != _selectionEnd) {
+ _selectionEnd = end;
+ _selectionMode = (_selectionStart != _selectionEnd ? PartialSelection : NoSelection);
+ update();
+ }
+ } else {
+ setFullSelection();
+ chatScene()->startGlobalSelection(this, event->pos());