source: trunk/src/gui/itemviews/qabstractitemview_p.h@ 642

Last change on this file since 642 was 561, checked in by Dmitry A. Kuminov, 15 years ago

trunk: Merged in qt 4.6.1 sources.

File size: 15.7 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
4** All rights reserved.
5** Contact: Nokia Corporation ([email protected])
6**
7** This file is part of the QtGui module of the Qt Toolkit.
8**
9** $QT_BEGIN_LICENSE:LGPL$
10** Commercial Usage
11** Licensees holding valid Qt Commercial licenses may use this file in
12** accordance with the Qt Commercial License Agreement provided with the
13** Software or, alternatively, in accordance with the terms contained in
14** a written agreement between you and Nokia.
15**
16** GNU Lesser General Public License Usage
17** Alternatively, this file may be used under the terms of the GNU Lesser
18** General Public License version 2.1 as published by the Free Software
19** Foundation and appearing in the file LICENSE.LGPL included in the
20** packaging of this file. Please review the following information to
21** ensure the GNU Lesser General Public License version 2.1 requirements
22** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
23**
24** In addition, as a special exception, Nokia gives you certain additional
25** rights. These rights are described in the Nokia Qt LGPL Exception
26** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
27**
28** GNU General Public License Usage
29** Alternatively, this file may be used under the terms of the GNU
30** General Public License version 3.0 as published by the Free Software
31** Foundation and appearing in the file LICENSE.GPL included in the
32** packaging of this file. Please review the following information to
33** ensure the GNU General Public License version 3.0 requirements will be
34** met: http://www.gnu.org/copyleft/gpl.html.
35**
36** If you have questions regarding the use of this file, please contact
37** Nokia at [email protected].
38** $QT_END_LICENSE$
39**
40****************************************************************************/
41
42#ifndef QABSTRACTITEMVIEW_P_H
43#define QABSTRACTITEMVIEW_P_H
44
45//
46// W A R N I N G
47// -------------
48//
49// This file is not part of the Qt API. It exists purely as an
50// implementation detail. This header file may change from version to
51// version without notice, or even be removed.
52//
53// We mean it.
54//
55
56#include "private/qabstractscrollarea_p.h"
57#include "private/qabstractitemmodel_p.h"
58#include "QtGui/qapplication.h"
59#include "QtCore/qdatetime.h"
60#include "QtGui/qevent.h"
61#include "QtGui/qmime.h"
62#include "QtGui/qpainter.h"
63#include "QtCore/qpair.h"
64#include "QtGui/qregion.h"
65#include "QtCore/qdebug.h"
66#include "QtGui/qpainter.h"
67#include "QtCore/qbasictimer.h"
68
69#ifndef QT_NO_ITEMVIEWS
70
71QT_BEGIN_NAMESPACE
72
73struct QEditorInfo
74{
75 QEditorInfo() : isStatic(false)
76 {
77 }
78
79 QEditorInfo(const QPersistentModelIndex &i, QWidget *e, bool b) : index(i), editor(e), isStatic(b)
80 {
81 }
82
83 QPersistentModelIndex index;
84 QPointer<QWidget> editor;
85 bool isStatic; //true when called from setIndexWidget
86
87};
88
89typedef QPair<QRect, QModelIndex> QItemViewPaintPair;
90typedef QList<QItemViewPaintPair> QItemViewPaintPairs;
91
92class QEmptyModel : public QAbstractItemModel
93{
94public:
95 explicit QEmptyModel(QObject *parent = 0) : QAbstractItemModel(parent) {}
96 QModelIndex index(int, int, const QModelIndex &) const { return QModelIndex(); }
97 QModelIndex parent(const QModelIndex &) const { return QModelIndex(); }
98 int rowCount(const QModelIndex &) const { return 0; }
99 int columnCount(const QModelIndex &) const { return 0; }
100 bool hasChildren(const QModelIndex &) const { return false; }
101 QVariant data(const QModelIndex &, int) const { return QVariant(); }
102};
103
104class Q_AUTOTEST_EXPORT QAbstractItemViewPrivate : public QAbstractScrollAreaPrivate
105{
106 Q_DECLARE_PUBLIC(QAbstractItemView)
107
108public:
109 QAbstractItemViewPrivate();
110 virtual ~QAbstractItemViewPrivate();
111
112 void init();
113
114 virtual void _q_rowsRemoved(const QModelIndex &parent, int start, int end);
115 virtual void _q_columnsAboutToBeRemoved(const QModelIndex &parent, int start, int end);
116 virtual void _q_columnsRemoved(const QModelIndex &parent, int start, int end);
117 virtual void _q_columnsInserted(const QModelIndex &parent, int start, int end);
118 virtual void _q_modelDestroyed();
119 virtual void _q_layoutChanged();
120 void _q_headerDataChanged() { doDelayedItemsLayout(); }
121
122 void fetchMore();
123
124 bool shouldEdit(QAbstractItemView::EditTrigger trigger, const QModelIndex &index) const;
125 bool shouldForwardEvent(QAbstractItemView::EditTrigger trigger, const QEvent *event) const;
126 bool shouldAutoScroll(const QPoint &pos) const;
127 void doDelayedItemsLayout(int delay = 0);
128 void interruptDelayedItemsLayout() const;
129
130 void startAutoScroll()
131 { // ### it would be nice to make this into a style hint one day
132 int scrollInterval = (verticalScrollMode == QAbstractItemView::ScrollPerItem) ? 150 : 50;
133 autoScrollTimer.start(scrollInterval, q_func());
134 autoScrollCount = 0;
135 }
136 void stopAutoScroll() { autoScrollTimer.stop(); autoScrollCount = 0;}
137
138
139 bool dropOn(QDropEvent *event, int *row, int *col, QModelIndex *index);
140 bool droppingOnItself(QDropEvent *event, const QModelIndex &index);
141
142 QWidget *editor(const QModelIndex &index, const QStyleOptionViewItem &options);
143 bool sendDelegateEvent(const QModelIndex &index, QEvent *event) const;
144 bool openEditor(const QModelIndex &index, QEvent *event);
145 void updateEditorData(const QModelIndex &topLeft, const QModelIndex &bottomRight);
146
147 QItemSelectionModel::SelectionFlags multiSelectionCommand(const QModelIndex &index,
148 const QEvent *event) const;
149 QItemSelectionModel::SelectionFlags extendedSelectionCommand(const QModelIndex &index,
150 const QEvent *event) const;
151 QItemSelectionModel::SelectionFlags contiguousSelectionCommand(const QModelIndex &index,
152 const QEvent *event) const;
153 virtual void selectAll(QItemSelectionModel::SelectionFlags command);
154
155 void setHoverIndex(const QPersistentModelIndex &index);
156
157 void checkMouseMove(const QPersistentModelIndex &index);
158 inline void checkMouseMove(const QPoint &pos) { checkMouseMove(q_func()->indexAt(pos)); }
159
160 inline QItemSelectionModel::SelectionFlags selectionBehaviorFlags() const
161 {
162 switch (selectionBehavior) {
163 case QAbstractItemView::SelectRows: return QItemSelectionModel::Rows;
164 case QAbstractItemView::SelectColumns: return QItemSelectionModel::Columns;
165 case QAbstractItemView::SelectItems: default: return QItemSelectionModel::NoUpdate;
166 }
167 }
168
169#ifndef QT_NO_DRAGANDDROP
170 virtual QAbstractItemView::DropIndicatorPosition position(const QPoint &pos, const QRect &rect, const QModelIndex &idx) const;
171
172 inline bool canDecode(QDropEvent *e) const {
173 QStringList modelTypes = model->mimeTypes();
174 const QMimeData *mime = e->mimeData();
175 for (int i = 0; i < modelTypes.count(); ++i)
176 if (mime->hasFormat(modelTypes.at(i))
177 && (e->dropAction() & model->supportedDropActions()))
178 return true;
179 return false;
180 }
181
182 inline void paintDropIndicator(QPainter *painter)
183 {
184 if (showDropIndicator && state == QAbstractItemView::DraggingState
185#ifndef QT_NO_CURSOR
186 && viewport->cursor().shape() != Qt::ForbiddenCursor
187#endif
188 ) {
189 QStyleOption opt;
190 opt.init(q_func());
191 opt.rect = dropIndicatorRect;
192 q_func()->style()->drawPrimitive(QStyle::PE_IndicatorItemViewItemDrop, &opt, painter, q_func());
193 }
194 }
195
196#endif
197 virtual QItemViewPaintPairs draggablePaintPairs(const QModelIndexList &indexes, QRect *r) const;
198
199 inline void releaseEditor(QWidget *editor) const {
200 if (editor) {
201 QObject::disconnect(editor, SIGNAL(destroyed(QObject*)),
202 q_func(), SLOT(editorDestroyed(QObject*)));
203 editor->removeEventFilter(itemDelegate);
204 editor->hide();
205 editor->deleteLater();
206 }
207 }
208
209 inline void executePostedLayout() const {
210 if (delayedPendingLayout && state != QAbstractItemView::CollapsingState) {
211 interruptDelayedItemsLayout();
212 const_cast<QAbstractItemView*>(q_func())->doItemsLayout();
213 }
214 }
215
216 inline void setDirtyRegion(const QRegion &visualRegion) {
217 updateRegion += visualRegion;
218 if (!updateTimer.isActive())
219 updateTimer.start(0, q_func());
220 }
221
222 inline void scrollDirtyRegion(int dx, int dy) {
223 scrollDelayOffset = QPoint(-dx, -dy);
224 updateDirtyRegion();
225 scrollDelayOffset = QPoint(0, 0);
226 }
227
228 inline void scrollContentsBy(int dx, int dy) {
229 scrollDirtyRegion(dx, dy);
230 viewport->scroll(dx, dy);
231 }
232
233 void updateDirtyRegion() {
234 updateTimer.stop();
235 viewport->update(updateRegion);
236 updateRegion = QRegion();
237 }
238
239 void clearOrRemove();
240 void checkPersistentEditorFocus();
241
242 QPixmap renderToPixmap(const QModelIndexList &indexes, QRect *r) const;
243
244 inline QPoint offset() const {
245 const Q_Q(QAbstractItemView);
246 return QPoint(q->isRightToLeft() ? -q->horizontalOffset()
247 : q->horizontalOffset(), q->verticalOffset());
248 }
249
250 QEditorInfo editorForIndex(const QModelIndex &index) const;
251 inline bool hasEditor(const QModelIndex &index) const {
252 return editorForIndex(index).editor != 0;
253 }
254
255 QModelIndex indexForEditor(QWidget *editor) const;
256 void addEditor(const QModelIndex &index, QWidget *editor, bool isStatic);
257 void removeEditor(QWidget *editor);
258
259 inline bool isAnimating() const {
260 return state == QAbstractItemView::AnimatingState;
261 }
262
263 inline QAbstractItemDelegate *delegateForIndex(const QModelIndex &index) const {
264 QAbstractItemDelegate *del;
265 if ((del = rowDelegates.value(index.row(), 0))) return del;
266 if ((del = columnDelegates.value(index.column(), 0))) return del;
267 return itemDelegate;
268 }
269
270 inline bool isIndexValid(const QModelIndex &index) const {
271 return (index.row() >= 0) && (index.column() >= 0) && (index.model() == model);
272 }
273 inline bool isIndexSelectable(const QModelIndex &index) const {
274 return (model->flags(index) & Qt::ItemIsSelectable);
275 }
276 inline bool isIndexEnabled(const QModelIndex &index) const {
277 return (model->flags(index) & Qt::ItemIsEnabled);
278 }
279 inline bool isIndexDropEnabled(const QModelIndex &index) const {
280 return (model->flags(index) & Qt::ItemIsDropEnabled);
281 }
282 inline bool isIndexDragEnabled(const QModelIndex &index) const {
283 return (model->flags(index) & Qt::ItemIsDragEnabled);
284 }
285
286 virtual bool selectionAllowed(const QModelIndex &index) const {
287 // in some views we want to go ahead with selections, even if the index is invalid
288 return isIndexValid(index) && isIndexSelectable(index);
289 }
290
291 // reimplemented from QAbstractScrollAreaPrivate
292 virtual QPoint contentsOffset() const {
293 Q_Q(const QAbstractItemView);
294 return QPoint(q->horizontalOffset(), q->verticalOffset());
295 }
296
297 /**
298 * For now, assume that we have few editors, if we need a more efficient implementation
299 * we should add a QMap<QAbstractItemDelegate*, int> member.
300 */
301 int delegateRefCount(const QAbstractItemDelegate *delegate) const
302 {
303 int ref = 0;
304 if (itemDelegate == delegate)
305 ++ref;
306
307 for (int maps = 0; maps < 2; ++maps) {
308 const QMap<int, QPointer<QAbstractItemDelegate> > *delegates = maps ? &columnDelegates : &rowDelegates;
309 for (QMap<int, QPointer<QAbstractItemDelegate> >::const_iterator it = delegates->begin();
310 it != delegates->end(); ++it) {
311 if (it.value() == delegate) {
312 ++ref;
313 // optimization, we are only interested in the ref count values 0, 1 or >=2
314 if (ref >= 2) {
315 return ref;
316 }
317 }
318 }
319 }
320 return ref;
321 }
322
323 /**
324 * return true if the index is registered as a QPersistentModelIndex
325 */
326 inline bool isPersistent(const QModelIndex &index) const
327 {
328 return static_cast<QAbstractItemModelPrivate *>(model->d_ptr.data())->persistent.indexes.contains(index);
329 }
330
331 QModelIndexList selectedDraggableIndexes() const;
332
333 QStyleOptionViewItemV4 viewOptionsV4() const;
334
335 void doDelayedReset()
336 {
337 //we delay the reset of the timer because some views (QTableView)
338 //with headers can't handle the fact that the model has been destroyed
339 //all _q_modelDestroyed slots must have been called
340 if (!delayedReset.isActive())
341 delayedReset.start(0, q_func());
342 }
343
344 QAbstractItemModel *model;
345 QPointer<QAbstractItemDelegate> itemDelegate;
346 QMap<int, QPointer<QAbstractItemDelegate> > rowDelegates;
347 QMap<int, QPointer<QAbstractItemDelegate> > columnDelegates;
348 QPointer<QItemSelectionModel> selectionModel;
349 QItemSelectionModel::SelectionFlag ctrlDragSelectionFlag;
350 bool noSelectionOnMousePress;
351
352 QAbstractItemView::SelectionMode selectionMode;
353 QAbstractItemView::SelectionBehavior selectionBehavior;
354
355 QList<QEditorInfo> editors;
356 QSet<QWidget*> persistent;
357 QWidget *currentlyCommittingEditor;
358
359 QPersistentModelIndex enteredIndex;
360 QPersistentModelIndex pressedIndex;
361 Qt::KeyboardModifiers pressedModifiers;
362 QPoint pressedPosition;
363 bool pressedAlreadySelected;
364
365 //forces the next mouseMoveEvent to send the viewportEntered signal
366 //if the mouse is over the viewport and not over an item
367 bool viewportEnteredNeeded;
368
369 QAbstractItemView::State state;
370 QAbstractItemView::EditTriggers editTriggers;
371 QAbstractItemView::EditTrigger lastTrigger;
372
373 QPersistentModelIndex root;
374 QPersistentModelIndex hover;
375
376 bool tabKeyNavigation;
377
378#ifndef QT_NO_DRAGANDDROP
379 bool showDropIndicator;
380 QRect dropIndicatorRect;
381 bool dragEnabled;
382 QAbstractItemView::DragDropMode dragDropMode;
383 bool overwrite;
384 QAbstractItemView::DropIndicatorPosition dropIndicatorPosition;
385 Qt::DropAction defaultDropAction;
386#endif
387
388#ifdef QT_SOFTKEYS_ENABLED
389 QAction *doneSoftKey;
390#endif
391
392 QString keyboardInput;
393 QTime keyboardInputTime;
394
395 bool autoScroll;
396 QBasicTimer autoScrollTimer;
397 int autoScrollMargin;
398 int autoScrollCount;
399 bool shouldScrollToCurrentOnShow; //used to know if we should scroll to current on show event
400 bool shouldClearStatusTip; //if there is a statustip currently shown that need to be cleared when leaving.
401
402 bool alternatingColors;
403
404 QSize iconSize;
405 Qt::TextElideMode textElideMode;
406
407 QRegion updateRegion; // used for the internal update system
408 QPoint scrollDelayOffset;
409
410 QBasicTimer updateTimer;
411 QBasicTimer delayedEditing;
412 QBasicTimer delayedAutoScroll; //used when an item is clicked
413 QBasicTimer delayedReset;
414
415 QAbstractItemView::ScrollMode verticalScrollMode;
416 QAbstractItemView::ScrollMode horizontalScrollMode;
417
418 bool currentIndexSet;
419
420 bool wrapItemText;
421 mutable bool delayedPendingLayout;
422
423private:
424 mutable QBasicTimer delayedLayout;
425 mutable QBasicTimer fetchMoreTimer;
426};
427
428QT_BEGIN_INCLUDE_NAMESPACE
429#include <qvector.h>
430QT_END_INCLUDE_NAMESPACE
431
432template <typename T>
433inline int qBinarySearch(const QVector<T> &vec, const T &item, int start, int end)
434{
435 int i = (start + end + 1) >> 1;
436 while (end - start > 0) {
437 if (vec.at(i) > item)
438 end = i - 1;
439 else
440 start = i;
441 i = (start + end + 1) >> 1;
442 }
443 return i;
444}
445
446QT_END_NAMESPACE
447
448#endif // QT_NO_ITEMVIEWS
449
450#endif // QABSTRACTITEMVIEW_P_H
Note: See TracBrowser for help on using the repository browser.