]> cloud.milkyroute.net Git - dolphin.git/blob - src/dolphiniconsview.cpp
The complexity of the class InformationPanel has grown a lot from KDE 4.0 to KDE...
[dolphin.git] / src / dolphiniconsview.cpp
1 /***************************************************************************
2 * Copyright (C) 2006 by Peter Penz (peter.penz@gmx.at) *
3 * *
4 * This program is free software; you can redistribute it and/or modify *
5 * it under the terms of the GNU General Public License as published by *
6 * the Free Software Foundation; either version 2 of the License, or *
7 * (at your option) any later version. *
8 * *
9 * This program is distributed in the hope that it will be useful, *
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
12 * GNU General Public License for more details. *
13 * *
14 * You should have received a copy of the GNU General Public License *
15 * along with this program; if not, write to the *
16 * Free Software Foundation, Inc., *
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
18 ***************************************************************************/
19
20 #include "dolphiniconsview.h"
21
22 #include "dolphincategorydrawer.h"
23 #include "dolphincontroller.h"
24 #include "settings/dolphinsettings.h"
25 #include "dolphinviewautoscroller.h"
26 #include "dolphin_iconsmodesettings.h"
27 #include "dolphin_generalsettings.h"
28 #include "draganddrophelper.h"
29 #include "selectionmanager.h"
30 #include "zoomlevelinfo.h"
31
32 #include <kcategorizedsortfilterproxymodel.h>
33 #include <kdialog.h>
34 #include <kdirmodel.h>
35 #include <kfileitemdelegate.h>
36
37 #include <QAbstractProxyModel>
38 #include <QApplication>
39 #include <QScrollBar>
40
41 DolphinIconsView::DolphinIconsView(QWidget* parent, DolphinController* controller) :
42 KCategorizedView(parent),
43 m_controller(controller),
44 m_selectionManager(0),
45 m_autoScroller(0),
46 m_categoryDrawer(0),
47 m_font(),
48 m_decorationSize(),
49 m_decorationPosition(QStyleOptionViewItem::Top),
50 m_displayAlignment(Qt::AlignHCenter),
51 m_itemSize(),
52 m_dropRect()
53 {
54 Q_ASSERT(controller != 0);
55 setLayoutDirection(Qt::LeftToRight);
56 setViewMode(QListView::IconMode);
57 setResizeMode(QListView::Adjust);
58 setMovement(QListView::Static);
59 setDragEnabled(true);
60 setEditTriggers(QAbstractItemView::NoEditTriggers);
61 viewport()->setAcceptDrops(true);
62
63 setMouseTracking(true);
64 m_autoScroller = new DolphinViewAutoScroller(this);
65
66 connect(this, SIGNAL(clicked(const QModelIndex&)),
67 controller, SLOT(requestTab(const QModelIndex&)));
68 if (KGlobalSettings::singleClick()) {
69 connect(this, SIGNAL(clicked(const QModelIndex&)),
70 controller, SLOT(triggerItem(const QModelIndex&)));
71 } else {
72 connect(this, SIGNAL(doubleClicked(const QModelIndex&)),
73 controller, SLOT(triggerItem(const QModelIndex&)));
74 }
75
76 if (DolphinSettings::instance().generalSettings()->showSelectionToggle()) {
77 m_selectionManager = new SelectionManager(this);
78 connect(m_selectionManager, SIGNAL(selectionChanged()),
79 this, SLOT(requestActivation()));
80 connect(m_controller, SIGNAL(urlChanged(const KUrl&)),
81 m_selectionManager, SLOT(reset()));
82 }
83
84 connect(this, SIGNAL(entered(const QModelIndex&)),
85 controller, SLOT(emitItemEntered(const QModelIndex&)));
86 connect(this, SIGNAL(viewportEntered()),
87 controller, SLOT(emitViewportEntered()));
88 connect(controller, SIGNAL(zoomLevelChanged(int)),
89 this, SLOT(setZoomLevel(int)));
90
91 const DolphinView* view = controller->dolphinView();
92 connect(view, SIGNAL(showPreviewChanged()),
93 this, SLOT(slotShowPreviewChanged()));
94 connect(view, SIGNAL(additionalInfoChanged()),
95 this, SLOT(slotAdditionalInfoChanged()));
96
97 // apply the icons mode settings to the widget
98 const IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
99 Q_ASSERT(settings != 0);
100
101 if (settings->useSystemFont()) {
102 m_font = KGlobalSettings::generalFont();
103 } else {
104 m_font = QFont(settings->fontFamily(),
105 settings->fontSize(),
106 settings->fontWeight(),
107 settings->italicFont());
108 }
109
110 setWordWrap(settings->numberOfTextlines() > 1);
111 updateGridSize(view->showPreview(), 0);
112
113 if (settings->arrangement() == QListView::TopToBottom) {
114 setFlow(QListView::LeftToRight);
115 m_decorationPosition = QStyleOptionViewItem::Top;
116 m_displayAlignment = Qt::AlignHCenter;
117 } else {
118 setFlow(QListView::TopToBottom);
119 m_decorationPosition = QStyleOptionViewItem::Left;
120 m_displayAlignment = Qt::AlignLeft | Qt::AlignVCenter;
121 }
122
123 m_categoryDrawer = new DolphinCategoryDrawer();
124 setCategoryDrawer(m_categoryDrawer);
125
126 setFocus();
127
128 connect(KGlobalSettings::self(), SIGNAL(settingsChanged(int)),
129 this, SLOT(slotGlobalSettingsChanged(int)));
130 }
131
132 DolphinIconsView::~DolphinIconsView()
133 {
134 delete m_categoryDrawer;
135 m_categoryDrawer = 0;
136 }
137
138 void DolphinIconsView::dataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight)
139 {
140 KCategorizedView::dataChanged(topLeft, bottomRight);
141
142 KCategorizedSortFilterProxyModel* proxyModel = dynamic_cast<KCategorizedSortFilterProxyModel*>(model());
143 if (!proxyModel->isCategorizedModel()) {
144 // bypass a QListView issue that items are not layout correctly if the decoration size of
145 // an index changes
146 scheduleDelayedItemsLayout();
147 }
148 }
149
150 QStyleOptionViewItem DolphinIconsView::viewOptions() const
151 {
152 QStyleOptionViewItem viewOptions = KCategorizedView::viewOptions();
153 viewOptions.font = m_font;
154 viewOptions.decorationPosition = m_decorationPosition;
155 viewOptions.decorationSize = m_decorationSize;
156 viewOptions.displayAlignment = m_displayAlignment;
157 viewOptions.showDecorationSelected = true;
158 return viewOptions;
159 }
160
161 void DolphinIconsView::contextMenuEvent(QContextMenuEvent* event)
162 {
163 KCategorizedView::contextMenuEvent(event);
164 m_controller->triggerContextMenuRequest(event->pos());
165 }
166
167 void DolphinIconsView::mousePressEvent(QMouseEvent* event)
168 {
169 m_controller->requestActivation();
170 const QModelIndex index = indexAt(event->pos());
171 if (index.isValid() && (event->button() == Qt::LeftButton)) {
172 // TODO: It should not be necessary to manually set the dragging state, but I could
173 // not reproduce this issue with a Qt-only example yet to find the root cause.
174 // Issue description: start Dolphin, split the view and drag an item from the
175 // inactive view to the active view by a very fast mouse movement. Result:
176 // the item gets selected instead of being dragged...
177 setState(QAbstractItemView::DraggingState);
178 }
179
180 if (!index.isValid()) {
181 if (QApplication::mouseButtons() & Qt::MidButton) {
182 m_controller->replaceUrlByClipboard();
183 }
184 const Qt::KeyboardModifiers modifier = QApplication::keyboardModifiers();
185 if (!(modifier & Qt::ShiftModifier) && !(modifier & Qt::ControlModifier)) {
186 clearSelection();
187 }
188 }
189
190 KCategorizedView::mousePressEvent(event);
191 }
192
193 void DolphinIconsView::startDrag(Qt::DropActions supportedActions)
194 {
195 // TODO: invoking KCategorizedView::startDrag() should not be necessary, we'll
196 // fix this in KDE 4.1
197 KCategorizedView::startDrag(supportedActions);
198 DragAndDropHelper::instance().startDrag(this, supportedActions, m_controller);
199 }
200
201 void DolphinIconsView::dragEnterEvent(QDragEnterEvent* event)
202 {
203 if (DragAndDropHelper::instance().isMimeDataSupported(event->mimeData())) {
204 event->acceptProposedAction();
205 }
206 }
207
208 void DolphinIconsView::dragLeaveEvent(QDragLeaveEvent* event)
209 {
210 KCategorizedView::dragLeaveEvent(event);
211 setDirtyRegion(m_dropRect);
212 }
213
214 void DolphinIconsView::dragMoveEvent(QDragMoveEvent* event)
215 {
216 KCategorizedView::dragMoveEvent(event);
217
218 // TODO: remove this code when the issue #160611 is solved in Qt 4.4
219 const QModelIndex index = indexAt(event->pos());
220 setDirtyRegion(m_dropRect);
221
222 m_dropRect.setSize(QSize()); // set as invalid
223 if (index.isValid()) {
224 const KFileItem item = m_controller->itemForIndex(index);
225 if (!item.isNull() && item.isDir()) {
226 m_dropRect = visualRect(index);
227 } else {
228 m_dropRect.setSize(QSize()); // set as invalid
229 }
230 }
231 if (DragAndDropHelper::instance().isMimeDataSupported(event->mimeData())) {
232 // accept url drops, independently from the destination item
233 event->acceptProposedAction();
234 }
235
236 setDirtyRegion(m_dropRect);
237 }
238
239 void DolphinIconsView::dropEvent(QDropEvent* event)
240 {
241 const QModelIndex index = indexAt(event->pos());
242 const KFileItem item = m_controller->itemForIndex(index);
243 m_controller->indicateDroppedUrls(item, m_controller->url(), event);
244
245 KCategorizedView::dropEvent(event);
246 }
247
248 QModelIndex DolphinIconsView::moveCursor(CursorAction cursorAction, Qt::KeyboardModifiers modifiers)
249 {
250 QModelIndex current = currentIndex();
251
252 QModelIndex newCurrent = QListView::moveCursor(cursorAction, modifiers);
253 if (newCurrent != current) {
254 return newCurrent;
255 }
256
257 // The cursor has not been moved by the base implementation. Provide a
258 // wrap behavior, so that the cursor will go to the next item when reaching
259 // the border.
260 const IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
261 if (settings->arrangement() == QListView::LeftToRight) {
262 switch (cursorAction) {
263 case MoveUp:
264 if (newCurrent.row() == 0) {
265 return newCurrent;
266 }
267 newCurrent = QListView::moveCursor(MoveLeft, modifiers);
268 selectionModel()->setCurrentIndex(newCurrent, QItemSelectionModel::NoUpdate);
269 newCurrent = QListView::moveCursor(MovePageDown, modifiers);
270 break;
271
272 case MoveDown:
273 if (newCurrent.row() == (model()->rowCount() - 1)) {
274 return newCurrent;
275 }
276 newCurrent = QListView::moveCursor(MovePageUp, modifiers);
277 selectionModel()->setCurrentIndex(newCurrent, QItemSelectionModel::Clear);
278 newCurrent = QListView::moveCursor(MoveRight, modifiers);
279 break;
280
281 default:
282 break;
283 }
284 } else {
285 switch (cursorAction) {
286 case MoveLeft:
287 if (newCurrent.row() == 0) {
288 return newCurrent;
289 }
290 newCurrent = QListView::moveCursor(MoveUp, modifiers);
291 do {
292 selectionModel()->setCurrentIndex(newCurrent, QItemSelectionModel::NoUpdate);
293 current = newCurrent;
294 newCurrent = QListView::moveCursor(MoveRight, modifiers);
295 } while (newCurrent != current);
296
297 if (!(modifiers & Qt::ControlModifier)) {
298 // Ctrl is not pressed -> selection is updated
299 if (!(modifiers & Qt::ShiftModifier)) {
300 // Shift is not pressed -> previous selection is lost
301 selectionModel()->clearSelection();
302 }
303 selectionModel()->setCurrentIndex(newCurrent, QItemSelectionModel::Select);
304 }
305 break;
306
307 case MoveRight:
308 if (newCurrent.row() == (model()->rowCount() - 1)) {
309 return newCurrent;
310 }
311 do {
312 selectionModel()->setCurrentIndex(newCurrent, QItemSelectionModel::NoUpdate);
313 current = newCurrent;
314 newCurrent = QListView::moveCursor(MoveLeft, modifiers);
315 } while (newCurrent != current);
316 newCurrent = QListView::moveCursor(MoveDown, modifiers);
317 break;
318
319 default:
320 break;
321 }
322 }
323
324 return newCurrent;
325 }
326
327 void DolphinIconsView::keyPressEvent(QKeyEvent* event)
328 {
329 KCategorizedView::keyPressEvent(event);
330 m_controller->handleKeyPressEvent(event);
331 }
332
333 void DolphinIconsView::wheelEvent(QWheelEvent* event)
334 {
335 if (m_selectionManager != 0) {
336 m_selectionManager->reset();
337 }
338
339 // let Ctrl+wheel events propagate to the DolphinView for icon zooming
340 if (event->modifiers() & Qt::ControlModifier) {
341 event->ignore();
342 return;
343 }
344
345 horizontalScrollBar()->setSingleStep(m_itemSize.width() / 10);
346 verticalScrollBar()->setSingleStep(m_itemSize.height() / 10);
347
348 KCategorizedView::wheelEvent(event);
349 // if the icons are aligned left to right, the vertical wheel event should
350 // be applied to the horizontal scrollbar
351 const IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
352 const bool scrollHorizontal = (event->orientation() == Qt::Vertical) &&
353 (settings->arrangement() == QListView::LeftToRight);
354 if (scrollHorizontal) {
355 QWheelEvent horizEvent(event->pos(),
356 event->delta(),
357 event->buttons(),
358 event->modifiers(),
359 Qt::Horizontal);
360 QApplication::sendEvent(horizontalScrollBar(), &horizEvent);
361 }
362 }
363
364 void DolphinIconsView::showEvent(QShowEvent* event)
365 {
366 KFileItemDelegate* delegate = dynamic_cast<KFileItemDelegate*>(itemDelegate());
367 delegate->setMaximumSize(m_itemSize);
368
369 KCategorizedView::showEvent(event);
370 }
371
372 void DolphinIconsView::leaveEvent(QEvent* event)
373 {
374 KCategorizedView::leaveEvent(event);
375 // if the mouse is above an item and moved very fast outside the widget,
376 // no viewportEntered() signal might be emitted although the mouse has been moved
377 // above the viewport
378 m_controller->emitViewportEntered();
379 }
380
381 void DolphinIconsView::currentChanged(const QModelIndex& current, const QModelIndex& previous)
382 {
383 KCategorizedView::currentChanged(current, previous);
384 m_autoScroller->handleCurrentIndexChange(current, previous);
385 }
386
387 void DolphinIconsView::resizeEvent(QResizeEvent* event)
388 {
389 KCategorizedView::resizeEvent(event);
390 const DolphinView* view = m_controller->dolphinView();
391 updateGridSize(view->showPreview(), view->additionalInfo().count());
392 }
393
394 void DolphinIconsView::slotShowPreviewChanged()
395 {
396 const DolphinView* view = m_controller->dolphinView();
397 updateGridSize(view->showPreview(), additionalInfoCount());
398 }
399
400 void DolphinIconsView::slotAdditionalInfoChanged()
401 {
402 const DolphinView* view = m_controller->dolphinView();
403 const bool showPreview = view->showPreview();
404 updateGridSize(showPreview, view->additionalInfo().count());
405 }
406
407 void DolphinIconsView::setZoomLevel(int level)
408 {
409 IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
410
411 const int oldIconSize = settings->iconSize();
412 int newIconSize = oldIconSize;
413
414 const bool showPreview = m_controller->dolphinView()->showPreview();
415 if (showPreview) {
416 const int previewSize = ZoomLevelInfo::iconSizeForZoomLevel(level);
417 settings->setPreviewSize(previewSize);
418 } else {
419 newIconSize = ZoomLevelInfo::iconSizeForZoomLevel(level);
420 settings->setIconSize(newIconSize);
421 }
422
423 // increase also the grid size
424 const int diff = newIconSize - oldIconSize;
425 settings->setItemWidth(settings->itemWidth() + diff);
426 settings->setItemHeight(settings->itemHeight() + diff);
427
428 updateGridSize(showPreview, additionalInfoCount());
429 }
430
431 void DolphinIconsView::requestActivation()
432 {
433 m_controller->requestActivation();
434 }
435
436 void DolphinIconsView::slotGlobalSettingsChanged(int category)
437 {
438 Q_UNUSED(category);
439
440 const IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
441 Q_ASSERT(settings != 0);
442 if (settings->useSystemFont()) {
443 m_font = KGlobalSettings::generalFont();
444 }
445
446 disconnect(this, SIGNAL(clicked(QModelIndex)), m_controller, SLOT(triggerItem(QModelIndex)));
447 disconnect(this, SIGNAL(doubleClicked(QModelIndex)), m_controller, SLOT(triggerItem(QModelIndex)));
448 if (KGlobalSettings::singleClick()) {
449 connect(this, SIGNAL(clicked(QModelIndex)), m_controller, SLOT(triggerItem(QModelIndex)));
450 } else {
451 connect(this, SIGNAL(doubleClicked(QModelIndex)), m_controller, SLOT(triggerItem(QModelIndex)));
452 }
453 }
454
455 void DolphinIconsView::updateGridSize(bool showPreview, int additionalInfoCount)
456 {
457 const IconsModeSettings* settings = DolphinSettings::instance().iconsModeSettings();
458 Q_ASSERT(settings != 0);
459
460 int itemWidth = settings->itemWidth();
461 int itemHeight = settings->itemHeight();
462 int size = settings->iconSize();
463
464 if (showPreview) {
465 const int previewSize = settings->previewSize();
466 const int diff = previewSize - size;
467 itemWidth += diff;
468 itemHeight += diff;
469
470 size = previewSize;
471 }
472
473 Q_ASSERT(additionalInfoCount >= 0);
474 itemHeight += additionalInfoCount * m_font.pointSize() * 2;
475
476 // Optimize the item size of the grid in a way to prevent large gaps on the
477 // right border (= row arrangement) or the bottom border (= column arrangement).
478 // There is no public API in QListView to find out the used width of the viewport
479 // for the layout. The following calculation of 'contentWidth'/'contentHeight'
480 // is based on QListViewPrivate::prepareItemsLayout() (Copyright (C) 2009 Nokia Corporation).
481 int frameAroundContents = 0;
482 if (style()->styleHint(QStyle::SH_ScrollView_FrameOnlyAroundContents)) {
483 frameAroundContents = style()->pixelMetric(QStyle::PM_DefaultFrameWidth) * 2;
484 }
485 const int spacing = settings->gridSpacing();
486 if (settings->arrangement() == QListView::TopToBottom) {
487 const int contentWidth = viewport()->width() - 1
488 - frameAroundContents
489 - style()->pixelMetric(QStyle::PM_ScrollBarExtent, 0, horizontalScrollBar());
490 const int gridWidth = itemWidth + spacing * 2;
491 const int horizItemCount = contentWidth / gridWidth;
492 if (horizItemCount > 0) {
493 itemWidth += (contentWidth - horizItemCount * gridWidth) / horizItemCount;
494 }
495
496 // The decoration width indirectly defines the maximum
497 // width for the text wrapping. To use the maximum item width
498 // for text wrapping, it is used as decoration width.
499 m_decorationSize = QSize(itemWidth, size);
500 setIconSize(QSize(itemWidth, size));
501 } else {
502 const int contentHeight = viewport()->height() - 1
503 - frameAroundContents
504 - style()->pixelMetric(QStyle::PM_ScrollBarExtent, 0, verticalScrollBar());
505 const int gridHeight = itemHeight + spacing;
506 const int vertItemCount = contentHeight / gridHeight;
507 if (vertItemCount > 0) {
508 itemHeight += (contentHeight - vertItemCount * gridHeight) / vertItemCount;
509 }
510
511 m_decorationSize = QSize(size, size);
512 setIconSize(QSize(size, size));
513 }
514
515 m_itemSize = QSize(itemWidth, itemHeight);
516 setGridSize(QSize(itemWidth + spacing * 2, itemHeight + spacing));
517
518 KFileItemDelegate* delegate = dynamic_cast<KFileItemDelegate*>(itemDelegate());
519 if (delegate != 0) {
520 delegate->setMaximumSize(m_itemSize);
521 }
522
523 if (m_selectionManager != 0) {
524 m_selectionManager->reset();
525 }
526 }
527
528 int DolphinIconsView::additionalInfoCount() const
529 {
530 const DolphinView* view = m_controller->dolphinView();
531 return view->additionalInfo().count();
532 }
533
534 #include "dolphiniconsview.moc"