/***************************************************************************
- * Copyright (C) 2005-08 by the Quassel Project *
+ * Copyright (C) 2005-2012 by the Quassel Project *
* devel@quassel-irc.org *
* *
* This program is free software; you can redistribute it and/or modify *
#define CHATSCENE_H_
#include <QAbstractItemModel>
+#include <QClipboard>
+#include <QGraphicsItem>
#include <QGraphicsScene>
#include <QSet>
+#include <QTimer>
-#include "columnhandleitem.h"
+#include "chatlinemodel.h"
#include "messagefilter.h"
class AbstractUiMsg;
class ChatItem;
class ChatLine;
+class ChatView;
+class ColumnHandleItem;
+class MarkerLineItem;
class WebPreviewItem;
class QGraphicsSceneMouseEvent;
SenderChatItemType,
ContentsChatItemType,
SearchHighlightType,
- WebPreviewType
+ WebPreviewType,
+ ColumnHandleType,
+ MarkerLineType
};
- ChatScene(QAbstractItemModel *model, const QString &idString, qreal width, QObject *parent);
+ enum ClickMode {
+ NoClick,
+ DragStartClick,
+ SingleClick,
+ DoubleClick,
+ TripleClick
+ };
+
+ ChatScene(QAbstractItemModel *model, const QString &idString, qreal width, ChatView *parent);
virtual ~ChatScene();
inline QAbstractItemModel *model() const { return _model; }
+ inline MessageFilter *filter() const { return qobject_cast<MessageFilter*>(_model); }
inline QString idString() const { return _idString; }
- int sectionByScenePos(int x);
- inline int sectionByScenePos(const QPoint &pos) { return sectionByScenePos(pos.x()); }
- inline bool isSingleBufferScene() const { return _singleBufferScene; }
- inline bool containsBuffer(const BufferId &id) const;
- inline ChatLine *chatLine(int row) { return (row < _lines.count()) ? _lines[row] : 0; }
+ int rowByScenePos(qreal y) const;
+ inline int rowByScenePos(const QPointF &pos) const { return rowByScenePos(pos.y()); }
+ ChatLineModel::ColumnType columnByScenePos(qreal x) const ;
+ inline ChatLineModel::ColumnType columnByScenePos(const QPointF &pos) const { return columnByScenePos(pos.x()); }
+
+ ChatView *chatView() const;
+ ChatItem *chatItemAt(const QPointF &pos) const;
+ inline ChatLine *chatLine(int row) const { return (row < _lines.count()) ? _lines.value(row) : 0; }
+ inline ChatLine *chatLine(const QModelIndex &index) const { return _lines.value(index.row()); }
+
+ //! Find the ChatLine belonging to a MsgId
+ /** Searches for the ChatLine belonging to a MsgId. If there are more than one ChatLine with the same msgId,
+ * the first one is returned.
+ * Note that this method performs a binary search, hence it has as complexity of O(log n).
+ * If matchExact is false, and we don't have an exact match for the given msgId, we return the visible line right
+ * above the requested one.
+ * \param msgId The message ID to look for
+ * \param matchExact Whether we find only exact matches
+ * \param ignoreDayChange Whether we ignore day change messages
+ * \return The ChatLine corresponding to the given MsgId
+ */
+ ChatLine *chatLine(MsgId msgId, bool matchExact = true, bool ignoreDayChange = true) const;
+
+ inline ChatLine *lastLine() const { return _lines.count() ? _lines.last() : 0; }
- inline ColumnHandleItem *firstColumnHandle() const { return _firstColHandle; }
- inline ColumnHandleItem *secondColumnHandle() const { return _secondColHandle; }
+ inline MarkerLineItem *markerLine() const { return _markerLine; }
+
+ inline bool isSingleBufferScene() const { return _singleBufferId.isValid(); }
+ inline BufferId singleBufferId() const { return _singleBufferId; }
+ bool containsBuffer(const BufferId &id) const;
+
+ ColumnHandleItem *firstColumnHandle() const;
+ ColumnHandleItem *secondColumnHandle() const;
inline CutoffMode senderCutoffMode() const { return _cutoffMode; }
inline void setSenderCutoffMode(CutoffMode mode) { _cutoffMode = mode; }
- bool isScrollingAllowed() const;
+ QString selection() const;
+ bool hasSelection() const;
+ bool hasGlobalSelection() const;
+ bool isPosOverSelection(const QPointF &) const;
+ bool isGloballySelecting() const;
+ void initiateDrag(QWidget *source);
- virtual bool event(QEvent *e);
+ bool isScrollingAllowed() const;
public slots:
void updateForViewport(qreal width, qreal height);
void setWidth(qreal width);
+ void layout(int start, int end, qreal width);
+
+ void setMarkerLineVisible(bool visible = true);
+ void setMarkerLine(MsgId msgId = MsgId());
+ void jumpToMarkerLine(bool requestBacklog);
// these are used by the chatitems to notify the scene and manage selections
void setSelectingItem(ChatItem *item);
ChatItem *selectingItem() const { return _selectingItem; }
void startGlobalSelection(ChatItem *item, const QPointF &itemPos);
- void putToClipboard(const QString &);
+ void clearGlobalSelection();
+ void clearSelection();
+ void selectionToClipboard(QClipboard::Mode = QClipboard::Clipboard);
+ void stringToClipboard(const QString &str, QClipboard::Mode = QClipboard::Clipboard);
void requestBacklog();
- void loadWebPreview(ChatItem *parentItem, const QString &url, const QRectF &urlRect);
+#ifdef HAVE_WEBKIT
+ void loadWebPreview(ChatItem *parentItem, const QUrl &url, const QRectF &urlRect);
void clearWebPreview(ChatItem *parentItem = 0);
+#endif
signals:
void lastLineChanged(QGraphicsItem *item, qreal offset);
void mouseMoveWhileSelecting(const QPointF &scenePos);
protected:
+ virtual void contextMenuEvent(QGraphicsSceneContextMenuEvent *contextMenuEvent);
virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *mouseEvent);
virtual void mousePressEvent(QGraphicsSceneMouseEvent *mouseEvent);
virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *mouseEvent);
+ virtual void mouseDoubleClickEvent(QGraphicsSceneMouseEvent *mouseEvent);
+ virtual void handleClick(Qt::MouseButton button, const QPointF &scenePos);
protected slots:
void rowsInserted(const QModelIndex &, int, int);
void rowsAboutToBeRemoved(const QModelIndex &, int, int);
+ void dataChanged(const QModelIndex &, const QModelIndex &);
private slots:
void firstHandlePositionChanged(qreal xpos);
void secondHandlePositionChanged(qreal xpos);
- void showWebPreviewEvent();
- void deleteWebPreviewEvent();
+#ifdef HAVE_WEBKIT
+ void webPreviewNextStep();
+#endif
void showWebPreviewChanged();
+ void rowsRemoved();
+
+ void clickTimeout();
+
private:
void setHandleXLimits();
void updateSelection(const QPointF &pos);
- QString selectionToString() const;
+ ChatView *_chatView;
QString _idString;
QAbstractItemModel *_model;
QList<ChatLine *> _lines;
- bool _singleBufferScene;
+ BufferId _singleBufferId;
// calls to QChatScene::sceneRect() are very expensive. As we manage the scenerect ourselves
// we store the size in a member variable.
void updateSceneRect(const QRectF &rect);
qreal _viewportHeight;
+ MarkerLineItem *_markerLine;
+ bool _markerLineVisible, _markerLineValid, _markerLineJumpPending;
+
ColumnHandleItem *_firstColHandle, *_secondColHandle;
qreal _firstColHandlePos, _secondColHandlePos;
CutoffMode _cutoffMode;
int _firstSelectionRow;
bool _isSelecting;
+ QTimer _clickTimer;
+ ClickMode _clickMode;
+ QPointF _clickPos;
+ bool _clickHandled;
+ bool _leftButtonPressed;
+
bool _showWebPreview;
+#ifdef HAVE_WEBKIT
struct WebPreview {
+ enum PreviewState {
+ NoPreview,
+ NewPreview,
+ DelayPreview,
+ ShowPreview,
+ HidePreview
+ };
ChatItem *parentItem;
QGraphicsItem *previewItem;
- QString url;
+ QUrl url;
QRectF urlRect;
- QTimer delayTimer;
- QTimer deleteTimer;
- WebPreview() : parentItem(0), previewItem(0) {}
+ PreviewState previewState;
+ QTimer timer;
+ WebPreview() : parentItem(0), previewItem(0), previewState(NoPreview) {}
};
WebPreview webPreview;
+#endif // HAVE_WEBKIT
};
-bool ChatScene::containsBuffer(const BufferId &id) const {
- MessageFilter *filter = qobject_cast<MessageFilter*>(model());
- if(filter)
- return filter->containsBuffer(id);
- else
- return false;
-}
-
#endif