Make the UI actually emit the required signal
[quassel.git] / src / qtui / chatscene.h
1 /***************************************************************************
2  *   Copyright (C) 2005-2016 by the Quassel Project                        *
3  *   devel@quassel-irc.org                                                 *
4  *                                                                         *
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.                                           *
9  *                                                                         *
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.                          *
14  *                                                                         *
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  *   51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.         *
19  ***************************************************************************/
20
21 #ifndef CHATSCENE_H_
22 #define CHATSCENE_H_
23
24 #include <QAbstractItemModel>
25 #include <QClipboard>
26 #include <QGraphicsItem>
27 #include <QGraphicsScene>
28 #include <QSet>
29 #include <QTimer>
30 #include <QUrl>
31
32 #include "chatlinemodel.h"
33 #include "messagefilter.h"
34
35 class AbstractUiMsg;
36 class ChatItem;
37 class ChatLine;
38 class ChatView;
39 class ColumnHandleItem;
40 class MarkerLineItem;
41 class WebPreviewItem;
42
43 class QGraphicsSceneMouseEvent;
44
45 class ChatScene : public QGraphicsScene
46 {
47     Q_OBJECT
48
49 public:
50     enum CutoffMode {
51         CutoffLeft,
52         CutoffRight
53     };
54
55     enum ItemType {
56         ChatLineType = QGraphicsItem::UserType + 1,
57         ChatItemType,
58         TimestampChatItemType,
59         SenderChatItemType,
60         ContentsChatItemType,
61         SearchHighlightType,
62         WebPreviewType,
63         ColumnHandleType,
64         MarkerLineType
65     };
66
67     enum ClickMode {
68         NoClick,
69         DragStartClick,
70         SingleClick,
71         DoubleClick,
72         TripleClick
73     };
74
75     ChatScene(QAbstractItemModel *model, const QString &idString, qreal width, ChatView *parent);
76     virtual ~ChatScene();
77
78     inline QAbstractItemModel *model() const { return _model; }
79     inline MessageFilter *filter() const { return qobject_cast<MessageFilter *>(_model); }
80     inline QString idString() const { return _idString; }
81
82     int rowByScenePos(qreal y) const;
83     inline int rowByScenePos(const QPointF &pos) const { return rowByScenePos(pos.y()); }
84     ChatLineModel::ColumnType columnByScenePos(qreal x) const;
85     inline ChatLineModel::ColumnType columnByScenePos(const QPointF &pos) const { return columnByScenePos(pos.x()); }
86
87     ChatView *chatView() const;
88     ChatItem *chatItemAt(const QPointF &pos) const;
89     inline ChatLine *chatLine(int row) const { return (row < _lines.count()) ? _lines.value(row) : 0; }
90     inline ChatLine *chatLine(const QModelIndex &index) const { return _lines.value(index.row()); }
91
92     //! Find the ChatLine belonging to a MsgId
93     /** Searches for the ChatLine belonging to a MsgId. If there are more than one ChatLine with the same msgId,
94      *  the first one is returned.
95      *  Note that this method performs a binary search, hence it has as complexity of O(log n).
96      *  If matchExact is false, and we don't have an exact match for the given msgId, we return the visible line right
97      *  above the requested one.
98      *  \param msgId      The message ID to look for
99      *  \param matchExact Whether we find only exact matches
100      *  \param ignoreDayChange Whether we ignore day change messages
101      *  \return The ChatLine corresponding to the given MsgId
102      */
103     ChatLine *chatLine(MsgId msgId, bool matchExact = true, bool ignoreDayChange = true) const;
104
105     inline ChatLine *lastLine() const { return _lines.count() ? _lines.last() : 0; }
106
107     inline MarkerLineItem *markerLine() const { return _markerLine; }
108
109     inline bool isSingleBufferScene() const { return _singleBufferId.isValid(); }
110     inline BufferId singleBufferId() const { return _singleBufferId; }
111     bool containsBuffer(const BufferId &id) const;
112
113     ColumnHandleItem *firstColumnHandle() const;
114     ColumnHandleItem *secondColumnHandle() const;
115
116     inline CutoffMode senderCutoffMode() const { return _cutoffMode; }
117     inline void setSenderCutoffMode(CutoffMode mode) { _cutoffMode = mode; }
118
119     QString selection() const;
120     bool hasSelection() const;
121     bool hasGlobalSelection() const;
122     bool isPosOverSelection(const QPointF &) const;
123     bool isGloballySelecting() const;
124     void initiateDrag(QWidget *source);
125
126     bool isScrollingAllowed() const;
127
128 public slots:
129     void updateForViewport(qreal width, qreal height);
130     void setWidth(qreal width);
131     void layout(int start, int end, qreal width);
132
133     void resetColumnWidths();
134
135     void setMarkerLineVisible(bool visible = true);
136     void setMarkerLine(MsgId msgId = MsgId());
137     void jumpToMarkerLine(bool requestBacklog);
138
139     // these are used by the chatitems to notify the scene and manage selections
140     void setSelectingItem(ChatItem *item);
141     ChatItem *selectingItem() const { return _selectingItem; }
142     void startGlobalSelection(ChatItem *item, const QPointF &itemPos);
143     void clearGlobalSelection();
144     void clearSelection();
145     void selectionToClipboard(QClipboard::Mode = QClipboard::Clipboard);
146     void stringToClipboard(const QString &str, QClipboard::Mode = QClipboard::Clipboard);
147
148     void webSearchOnSelection();
149
150     void requestBacklog();
151
152 #if defined HAVE_WEBKIT || defined HAVE_WEBENGINE
153     void loadWebPreview(ChatItem *parentItem, const QUrl &url, const QRectF &urlRect);
154     void clearWebPreview(ChatItem *parentItem = 0);
155 #endif
156
157 signals:
158     void lastLineChanged(QGraphicsItem *item, qreal offset);
159     void layoutChanged(); // indicates changes to the scenerect due to resizing of the contentsitems
160     void mouseMoveWhileSelecting(const QPointF &scenePos);
161
162 protected:
163     virtual void contextMenuEvent(QGraphicsSceneContextMenuEvent *contextMenuEvent);
164     virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *mouseEvent);
165     virtual void mousePressEvent(QGraphicsSceneMouseEvent *mouseEvent);
166     virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *mouseEvent);
167     virtual void mouseDoubleClickEvent(QGraphicsSceneMouseEvent *mouseEvent);
168     virtual void handleClick(Qt::MouseButton button, const QPointF &scenePos);
169
170 protected slots:
171     void rowsInserted(const QModelIndex &, int, int);
172     void rowsAboutToBeRemoved(const QModelIndex &, int, int);
173     void dataChanged(const QModelIndex &, const QModelIndex &);
174
175 private slots:
176     void firstHandlePositionChanged(qreal xpos);
177     void secondHandlePositionChanged(qreal xpos);
178 #if defined HAVE_WEBKIT || defined HAVE_WEBENGINE
179     void webPreviewNextStep();
180 #endif
181     void showWebPreviewChanged();
182
183     /**
184      * Updates the local setting cache of whether or not to show sender brackets
185      */
186     void showSenderBracketsChanged();
187
188     /**
189      * Updates the local setting cache of whether or not to use the custom timestamp format
190      */
191     void useCustomTimestampFormatChanged();
192
193     /**
194      * Updates the local setting cache of the timestamp format string
195      */
196     void timestampFormatStringChanged();
197
198     /**
199      * Updates the status of whether or not the timestamp format string contains brackets
200      *
201      * When the timestamp contains brackets -and- showSenderBrackets is disabled, we need to
202      * automatically add brackets.  This function checks if the timestamp has brackets and stores
203      * the result, rather than checking each time text is copied.
204      */
205     void updateTimestampHasBrackets();
206
207     void rowsRemoved();
208
209     void clickTimeout();
210
211 private:
212     void setHandleXLimits();
213     void updateSelection(const QPointF &pos);
214
215     ChatView *_chatView;
216     QString _idString;
217     QAbstractItemModel *_model;
218     QList<ChatLine *> _lines;
219     BufferId _singleBufferId;
220
221     // calls to QChatScene::sceneRect() are very expensive. As we manage the scenerect ourselves
222     // we store the size in a member variable.
223     QRectF _sceneRect;
224     int _firstLineRow; // the first row to display (aka: not a daychange msg)
225     void updateSceneRect(qreal width);
226     inline void updateSceneRect() { updateSceneRect(_sceneRect.width()); }
227     void updateSceneRect(const QRectF &rect);
228     qreal _viewportHeight;
229
230     MarkerLineItem *_markerLine;
231     bool _markerLineVisible, _markerLineValid, _markerLineJumpPending;
232
233     ColumnHandleItem *_firstColHandle, *_secondColHandle;
234     qreal _firstColHandlePos, _secondColHandlePos;
235     int _defaultFirstColHandlePos, _defaultSecondColHandlePos;
236     CutoffMode _cutoffMode;
237
238     ChatItem *_selectingItem;
239     int _selectionStartCol, _selectionMinCol;
240     int _selectionStart;
241     int _selectionEnd;
242     int _firstSelectionRow;
243     bool _isSelecting;
244
245     QTimer _clickTimer;
246     ClickMode _clickMode;
247     QPointF _clickPos;
248     bool _clickHandled;
249     bool _leftButtonPressed;
250
251     bool _showWebPreview;
252
253     bool _showSenderBrackets;  /// If true, show brackets around sender names
254
255     bool _useCustomTimestampFormat; /// If true, use the custom timestamp format
256     QString _timestampFormatString; /// Format of the timestamp string
257     bool _timestampHasBrackets;     /// If true, timestamp format has [brackets] of some sort
258
259     static const int _webSearchSelectionTextMaxVisible = 24;
260
261 #if defined HAVE_WEBKIT || defined HAVE_WEBENGINE
262     struct WebPreview {
263         enum PreviewState {
264             NoPreview,
265             NewPreview,
266             DelayPreview,
267             ShowPreview,
268             HidePreview
269         };
270         ChatItem *parentItem;
271         QGraphicsItem *previewItem;
272         QUrl url;
273         QRectF urlRect;
274         PreviewState previewState;
275         QTimer timer;
276         WebPreview() : parentItem(0), previewItem(0), previewState(NoPreview) {}
277     };
278     WebPreview webPreview;
279 #endif // HAVE_WEBKIT || HAVE_WEBENGINE
280 };
281
282
283 #endif