]> cloud.milkyroute.net Git - dolphin.git/blob - src/dolphinview.cpp
include cleanup
[dolphin.git] / src / dolphinview.cpp
1 /***************************************************************************
2 * Copyright (C) 2006 by Peter Penz <peter.penz@gmx.at> *
3 * Copyright (C) 2006 by Gregor Kališnik <gregor@podnapisi.net> *
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) any later version. *
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 #include "dolphinview.h"
22
23 #include <QtGui/QApplication>
24 #include <QtGui/QClipboard>
25 #include <QtGui/QKeyEvent>
26 #include <QtGui/QItemSelection>
27 #include <QtGui/QBoxLayout>
28 #include <QtCore/QTimer>
29 #include <QtGui/QScrollBar>
30
31 #include <kdirmodel.h>
32 #include <kfileitemdelegate.h>
33 #include <kfileplacesmodel.h>
34 #include <klocale.h>
35 #include <kiconeffect.h>
36 #include <kio/netaccess.h>
37 #include <kio/renamedialog.h>
38 #include <kio/previewjob.h>
39 #include <kmimetyperesolver.h>
40 #include <konqmimedata.h>
41 #include <konq_operations.h>
42 #include <kurl.h>
43
44 #include "dolphincolumnview.h"
45 #include "dolphincontroller.h"
46 #include "dolphinstatusbar.h"
47 #include "dolphinmainwindow.h"
48 #include "dolphindirlister.h"
49 #include "dolphinsortfilterproxymodel.h"
50 #include "dolphindetailsview.h"
51 #include "dolphiniconsview.h"
52 #include "dolphincontextmenu.h"
53 #include "dolphinitemcategorizer.h"
54 #include "filterbar.h"
55 #include "renamedialog.h"
56 #include "kurlnavigator.h"
57 #include "viewproperties.h"
58 #include "dolphinsettings.h"
59 #include "dolphin_generalsettings.h"
60
61 DolphinView::DolphinView(DolphinMainWindow* mainWindow,
62 QWidget* parent,
63 const KUrl& url,
64 Mode mode,
65 bool showHiddenFiles) :
66 QWidget(parent),
67 m_showProgress(false),
68 m_blockContentsMovedSignal(false),
69 m_initializeColumnView(false),
70 m_mode(mode),
71 m_iconSize(0),
72 m_folderCount(0),
73 m_fileCount(0),
74 m_mainWindow(mainWindow),
75 m_topLayout(0),
76 m_urlNavigator(0),
77 m_controller(0),
78 m_iconsView(0),
79 m_detailsView(0),
80 m_columnView(0),
81 m_fileItemDelegate(0),
82 m_filterBar(0),
83 m_statusBar(0),
84 m_dirModel(0),
85 m_dirLister(0),
86 m_proxyModel(0)
87 {
88 hide();
89 setFocusPolicy(Qt::StrongFocus);
90 m_topLayout = new QVBoxLayout(this);
91 m_topLayout->setSpacing(0);
92 m_topLayout->setMargin(0);
93
94 connect(m_mainWindow, SIGNAL(activeViewChanged()),
95 this, SLOT(updateActivationState()));
96
97 QClipboard* clipboard = QApplication::clipboard();
98 connect(clipboard, SIGNAL(dataChanged()),
99 this, SLOT(updateCutItems()));
100
101 m_urlNavigator = new KUrlNavigator(DolphinSettings::instance().placesModel(), url, this);
102 m_urlNavigator->setUrlEditable(DolphinSettings::instance().generalSettings()->editableUrl());
103 m_urlNavigator->setHomeUrl(DolphinSettings::instance().generalSettings()->homeUrl());
104 connect(m_urlNavigator, SIGNAL(urlChanged(const KUrl&)),
105 this, SLOT(changeDirectory(const KUrl&)));
106 connect(m_urlNavigator, SIGNAL(urlsDropped(const KUrl::List&, const KUrl&)),
107 this, SLOT(dropUrls(const KUrl::List&, const KUrl&)));
108 connect(m_urlNavigator, SIGNAL(activated()),
109 this, SLOT(requestActivation()));
110 connect(this, SIGNAL(contentsMoved(int, int)),
111 m_urlNavigator, SLOT(savePosition(int, int)));
112
113 m_statusBar = new DolphinStatusBar(this);
114
115 m_dirLister = new DolphinDirLister();
116 m_dirLister->setAutoUpdate(true);
117 m_dirLister->setMainWindow(this);
118 m_dirLister->setShowingDotFiles(showHiddenFiles);
119 m_dirLister->setDelayedMimeTypes(true);
120
121 connect(m_dirLister, SIGNAL(clear()),
122 this, SLOT(updateStatusBar()));
123 connect(m_dirLister, SIGNAL(percent(int)),
124 this, SLOT(updateProgress(int)));
125 connect(m_dirLister, SIGNAL(deleteItem(KFileItem*)),
126 this, SLOT(updateStatusBar()));
127 connect(m_dirLister, SIGNAL(completed()),
128 this, SLOT(updateItemCount()));
129 connect(m_dirLister, SIGNAL(completed()),
130 this, SLOT(updateCutItems()));
131 connect(m_dirLister, SIGNAL(newItems(const KFileItemList&)),
132 this, SLOT(generatePreviews(const KFileItemList&)));
133 connect(m_dirLister, SIGNAL(infoMessage(const QString&)),
134 this, SLOT(showInfoMessage(const QString&)));
135 connect(m_dirLister, SIGNAL(errorMessage(const QString&)),
136 this, SLOT(showErrorMessage(const QString&)));
137
138 m_dirModel = new KDirModel();
139 m_dirModel->setDirLister(m_dirLister);
140 m_dirModel->setDropsAllowed(KDirModel::DropOnDirectory);
141
142 m_proxyModel = new DolphinSortFilterProxyModel(this);
143 m_proxyModel->setSourceModel(m_dirModel);
144
145 m_controller = new DolphinController(this);
146 connect(m_controller, SIGNAL(requestContextMenu(const QPoint&)),
147 this, SLOT(openContextMenu(const QPoint&)));
148 connect(m_controller, SIGNAL(urlsDropped(const KUrl::List&, const QModelIndex&, QWidget*)),
149 this, SLOT(dropUrls(const KUrl::List&, const QModelIndex&, QWidget*)));
150 connect(m_controller, SIGNAL(sortingChanged(DolphinView::Sorting)),
151 this, SLOT(updateSorting(DolphinView::Sorting)));
152 connect(m_controller, SIGNAL(sortOrderChanged(Qt::SortOrder)),
153 this, SLOT(updateSortOrder(Qt::SortOrder)));
154 connect(m_controller, SIGNAL(itemTriggered(const QModelIndex&)),
155 this, SLOT(triggerItem(const QModelIndex&)));
156 connect(m_controller, SIGNAL(selectionChanged()),
157 this, SLOT(emitSelectionChangedSignal()));
158 connect(m_controller, SIGNAL(activated()),
159 this, SLOT(requestActivation()));
160
161 createView();
162
163 m_iconSize = K3Icon::SizeMedium;
164
165 m_filterBar = new FilterBar(this);
166 m_filterBar->hide();
167 connect(m_filterBar, SIGNAL(filterChanged(const QString&)),
168 this, SLOT(changeNameFilter(const QString&)));
169 connect(m_filterBar, SIGNAL(closeRequest()),
170 this, SLOT(closeFilterBar()));
171
172 m_topLayout->addWidget(m_urlNavigator);
173 m_topLayout->addWidget(itemView());
174 m_topLayout->addWidget(m_filterBar);
175 m_topLayout->addWidget(m_statusBar);
176 }
177
178 DolphinView::~DolphinView()
179 {
180 delete m_dirLister;
181 m_dirLister = 0;
182 }
183
184 void DolphinView::setUrl(const KUrl& url)
185 {
186 m_urlNavigator->setUrl(url);
187 m_controller->setUrl(url);
188 }
189
190 const KUrl& DolphinView::url() const
191 {
192 return m_urlNavigator->url();
193 }
194
195 bool DolphinView::isActive() const
196 {
197 return m_mainWindow->activeView() == this;
198 }
199
200 void DolphinView::setMode(Mode mode)
201 {
202 if (mode == m_mode) {
203 return; // the wished mode is already set
204 }
205
206 m_mode = mode;
207
208 if (isColumnViewActive()) {
209 // When changing the mode in the column view, it makes sense
210 // to go back to the root URL of the column view automatically.
211 // Otherwise there it would not be possible to turn off the column view
212 // without focusing the first column.
213 setUrl(m_dirLister->url());
214 }
215
216 ViewProperties props(m_urlNavigator->url());
217 props.setViewMode(m_mode);
218
219 createView();
220 startDirLister(m_urlNavigator->url());
221
222 emit modeChanged();
223 }
224
225 DolphinView::Mode DolphinView::mode() const
226 {
227 return m_mode;
228 }
229
230 void DolphinView::setShowPreview(bool show)
231 {
232 ViewProperties props(m_urlNavigator->url());
233 props.setShowPreview(show);
234
235 m_controller->setShowPreview(show);
236 emit showPreviewChanged();
237
238 startDirLister(m_urlNavigator->url(), true);
239 }
240
241 bool DolphinView::showPreview() const
242 {
243 return m_controller->showPreview();
244 }
245
246 void DolphinView::setShowHiddenFiles(bool show)
247 {
248 if (m_dirLister->showingDotFiles() == show) {
249 return;
250 }
251
252 ViewProperties props(m_urlNavigator->url());
253 props.setShowHiddenFiles(show);
254
255 m_dirLister->setShowingDotFiles(show);
256 emit showHiddenFilesChanged();
257
258 startDirLister(m_urlNavigator->url(), true);
259 }
260
261 bool DolphinView::showHiddenFiles() const
262 {
263 return m_dirLister->showingDotFiles();
264 }
265
266 void DolphinView::setCategorizedSorting(bool categorized)
267 {
268 if (!supportsCategorizedSorting() || (categorized == categorizedSorting())) {
269 return;
270 }
271
272 Q_ASSERT(m_iconsView != 0);
273 if (categorized) {
274 Q_ASSERT(m_iconsView->itemCategorizer() == 0);
275 m_iconsView->setItemCategorizer(new DolphinItemCategorizer());
276 } else {
277 KItemCategorizer* categorizer = m_iconsView->itemCategorizer();
278 m_iconsView->setItemCategorizer(0);
279 delete categorizer;
280 }
281
282 ViewProperties props(m_urlNavigator->url());
283 props.setCategorizedSorting(categorized);
284 props.save();
285
286 emit categorizedSortingChanged();
287 }
288
289 bool DolphinView::categorizedSorting() const
290 {
291 if (!supportsCategorizedSorting()) {
292 return false;
293 }
294
295 Q_ASSERT(m_iconsView != 0);
296 return m_iconsView->itemCategorizer() != 0;
297 }
298
299 bool DolphinView::supportsCategorizedSorting() const
300 {
301 return m_iconsView != 0;
302 }
303
304 void DolphinView::renameSelectedItems()
305 {
306 DolphinView* view = mainWindow()->activeView();
307 const KUrl::List urls = selectedUrls();
308 if (urls.count() > 1) {
309 // More than one item has been selected for renaming. Open
310 // a rename dialog and rename all items afterwards.
311 RenameDialog dialog(urls);
312 if (dialog.exec() == QDialog::Rejected) {
313 return;
314 }
315
316 const QString& newName = dialog.newName();
317 if (newName.isEmpty()) {
318 view->statusBar()->setMessage(dialog.errorString(),
319 DolphinStatusBar::Error);
320 } else {
321 // TODO: check how this can be integrated into KonqUndoManager/KonqOperations
322 // as one operation instead of n rename operations like it is done now...
323 Q_ASSERT(newName.contains('#'));
324
325 // iterate through all selected items and rename them...
326 const int replaceIndex = newName.indexOf('#');
327 Q_ASSERT(replaceIndex >= 0);
328 int index = 1;
329
330 KUrl::List::const_iterator it = urls.begin();
331 KUrl::List::const_iterator end = urls.end();
332 while (it != end) {
333 const KUrl& oldUrl = *it;
334 QString number;
335 number.setNum(index++);
336
337 QString name(newName);
338 name.replace(replaceIndex, 1, number);
339
340 if (oldUrl.fileName() != name) {
341 KUrl newUrl = oldUrl;
342 newUrl.setFileName(name);
343 m_mainWindow->rename(oldUrl, newUrl);
344 }
345 ++it;
346 }
347 }
348 } else {
349 // Only one item has been selected for renaming. Use the custom
350 // renaming mechanism from the views.
351 Q_ASSERT(urls.count() == 1);
352
353 // TODO: Think about using KFileItemDelegate as soon as it supports editing.
354 // Currently the RenameDialog is used, but I'm not sure whether inline renaming
355 // is a benefit for the user at all -> let's wait for some input first...
356 RenameDialog dialog(urls);
357 if (dialog.exec() == QDialog::Rejected) {
358 return;
359 }
360
361 const QString& newName = dialog.newName();
362 if (newName.isEmpty()) {
363 view->statusBar()->setMessage(dialog.errorString(),
364 DolphinStatusBar::Error);
365 } else {
366 const KUrl& oldUrl = urls.first();
367 KUrl newUrl = oldUrl;
368 newUrl.setFileName(newName);
369 m_mainWindow->rename(oldUrl, newUrl);
370 }
371 }
372 }
373
374 void DolphinView::selectAll()
375 {
376 selectAll(QItemSelectionModel::Select);
377 }
378
379 void DolphinView::invertSelection()
380 {
381 selectAll(QItemSelectionModel::Toggle);
382 }
383
384 DolphinStatusBar* DolphinView::statusBar() const
385 {
386 return m_statusBar;
387 }
388
389 int DolphinView::contentsX() const
390 {
391 return itemView()->horizontalScrollBar()->value();
392 }
393
394 int DolphinView::contentsY() const
395 {
396 return itemView()->verticalScrollBar()->value();
397 }
398
399 void DolphinView::emitRequestItemInfo(const KUrl& url)
400 {
401 emit requestItemInfo(url);
402 }
403
404 bool DolphinView::isFilterBarVisible() const
405 {
406 return m_filterBar->isVisible();
407 }
408
409 bool DolphinView::isUrlEditable() const
410 {
411 return m_urlNavigator->isUrlEditable();
412 }
413
414 void DolphinView::zoomIn()
415 {
416 m_controller->triggerZoomIn();
417 }
418
419 void DolphinView::zoomOut()
420 {
421 m_controller->triggerZoomOut();
422 }
423
424 bool DolphinView::isZoomInPossible() const
425 {
426 return m_controller->isZoomInPossible();
427 }
428
429 bool DolphinView::isZoomOutPossible() const
430 {
431 return m_controller->isZoomOutPossible();
432 }
433
434 void DolphinView::setSorting(Sorting sorting)
435 {
436 if (sorting != this->sorting()) {
437 updateSorting(sorting);
438 }
439 }
440
441 DolphinView::Sorting DolphinView::sorting() const
442 {
443 return m_proxyModel->sorting();
444 }
445
446 void DolphinView::setSortOrder(Qt::SortOrder order)
447 {
448 if (sortOrder() != order) {
449 updateSortOrder(order);
450 }
451 }
452
453 Qt::SortOrder DolphinView::sortOrder() const
454 {
455 return m_proxyModel->sortOrder();
456 }
457
458 void DolphinView::setAdditionalInfo(KFileItemDelegate::AdditionalInformation info)
459 {
460 ViewProperties props(m_urlNavigator->url());
461 props.setAdditionalInfo(info);
462
463 m_fileItemDelegate->setAdditionalInformation(info);
464
465 emit additionalInfoChanged(info);
466 startDirLister(m_urlNavigator->url(), true);
467 }
468
469 KFileItemDelegate::AdditionalInformation DolphinView::additionalInfo() const
470 {
471 return m_fileItemDelegate->additionalInformation();
472 }
473
474 void DolphinView::goBack()
475 {
476 m_urlNavigator->goBack();
477 }
478
479 void DolphinView::goForward()
480 {
481 m_urlNavigator->goForward();
482 }
483
484 void DolphinView::goUp()
485 {
486 m_urlNavigator->goUp();
487 }
488
489 void DolphinView::goHome()
490 {
491 m_urlNavigator->goHome();
492 }
493
494 void DolphinView::setUrlEditable(bool editable)
495 {
496 m_urlNavigator->setUrlEditable(editable);
497 }
498
499 bool DolphinView::hasSelection() const
500 {
501 return itemView()->selectionModel()->hasSelection();
502 }
503
504 void DolphinView::clearSelection()
505 {
506 itemView()->selectionModel()->clear();
507 }
508
509 KFileItemList DolphinView::selectedItems() const
510 {
511 const QAbstractItemView* view = itemView();
512
513 // Our view has a selection, we will map them back to the DirModel
514 // and then fill the KFileItemList.
515 Q_ASSERT((view != 0) && (view->selectionModel() != 0));
516
517 const QItemSelection selection = m_proxyModel->mapSelectionToSource(view->selectionModel()->selection());
518 KFileItemList itemList;
519
520 const QModelIndexList indexList = selection.indexes();
521 QModelIndexList::const_iterator end = indexList.end();
522 for (QModelIndexList::const_iterator it = indexList.begin(); it != end; ++it) {
523 Q_ASSERT((*it).isValid());
524
525 KFileItem* item = m_dirModel->itemForIndex(*it);
526 if (item != 0) {
527 itemList.append(item);
528 }
529 }
530
531 return itemList;
532 }
533
534 KUrl::List DolphinView::selectedUrls() const
535 {
536 KUrl::List urls;
537
538 const KFileItemList list = selectedItems();
539 KFileItemList::const_iterator it = list.begin();
540 const KFileItemList::const_iterator end = list.end();
541 while (it != end) {
542 KFileItem* item = *it;
543 urls.append(item->url());
544 ++it;
545 }
546
547 return urls;
548 }
549
550 KFileItem* DolphinView::fileItem(const QModelIndex index) const
551 {
552 const QModelIndex dirModelIndex = m_proxyModel->mapToSource(index);
553 return m_dirModel->itemForIndex(dirModelIndex);
554 }
555
556 void DolphinView::rename(const KUrl& source, const QString& newName)
557 {
558 bool ok = false;
559
560 if (newName.isEmpty() || (source.fileName() == newName)) {
561 return;
562 }
563
564 KUrl dest(source.upUrl());
565 dest.addPath(newName);
566
567 const bool destExists = KIO::NetAccess::exists(dest,
568 false,
569 mainWindow()->activeView());
570 if (destExists) {
571 // the destination already exists, hence ask the user
572 // how to proceed...
573 KIO::RenameDialog renameDialog(this,
574 i18n("File Already Exists"),
575 source.path(),
576 dest.path(),
577 KIO::M_OVERWRITE);
578 switch (renameDialog.exec()) {
579 case KIO::R_OVERWRITE:
580 // the destination should be overwritten
581 ok = KIO::NetAccess::file_move(source, dest, -1, true);
582 break;
583
584 case KIO::R_RENAME: {
585 // a new name for the destination has been used
586 KUrl newDest(renameDialog.newDestUrl());
587 ok = KIO::NetAccess::file_move(source, newDest);
588 break;
589 }
590
591 default:
592 // the renaming operation has been canceled
593 return;
594 }
595 } else {
596 // no destination exists, hence just move the file to
597 // do the renaming
598 ok = KIO::NetAccess::file_move(source, dest);
599 }
600
601 const QString destFileName = dest.fileName();
602 if (ok) {
603 m_statusBar->setMessage(i18n("Renamed file '%1' to '%2'.", source.fileName(), destFileName),
604 DolphinStatusBar::OperationCompleted);
605
606 KonqOperations::rename(this, source, destFileName);
607 } else {
608 m_statusBar->setMessage(i18n("Renaming of file '%1' to '%2' failed.", source.fileName(), destFileName),
609 DolphinStatusBar::Error);
610 }
611 }
612
613 void DolphinView::reload()
614 {
615 const KUrl& url = m_urlNavigator->url();
616 changeDirectory(url);
617 startDirLister(url, true);
618 }
619
620 void DolphinView::mouseReleaseEvent(QMouseEvent* event)
621 {
622 QWidget::mouseReleaseEvent(event);
623 mainWindow()->setActiveView(this);
624 }
625
626 DolphinMainWindow* DolphinView::mainWindow() const
627 {
628 return m_mainWindow;
629 }
630
631 void DolphinView::changeDirectory(const KUrl& url)
632 {
633 if (!isActive()) {
634 requestActivation();
635 }
636
637 const ViewProperties props(url);
638
639 const Mode mode = props.viewMode();
640 bool changeMode = (m_mode != mode);
641 if (changeMode && isColumnViewActive()) {
642 // The column view is active. Only change the
643 // mode if the current URL is no child of the column view.
644 if (m_dirLister->url().isParentOf(url)) {
645 changeMode = false;
646 }
647 }
648
649 if (changeMode) {
650 m_mode = mode;
651 createView();
652 emit modeChanged();
653
654 if (m_mode == ColumnView) {
655 // The mode has been changed to the Column View. When starting the dir
656 // lister with DolphinView::startDirLister() it is important to give a
657 // hint that the dir lister may not keep the current directory
658 // although this is the default for showing a hierarchy.
659 m_initializeColumnView = true;
660 }
661 }
662
663 const bool showHiddenFiles = props.showHiddenFiles();
664 if (showHiddenFiles != m_dirLister->showingDotFiles()) {
665 m_dirLister->setShowingDotFiles(showHiddenFiles);
666 emit showHiddenFilesChanged();
667 }
668
669 const bool categorized = props.categorizedSorting();
670 if (categorized != categorizedSorting()) {
671 if (supportsCategorizedSorting()) {
672 Q_ASSERT(m_iconsView != 0);
673 if (categorized) {
674 Q_ASSERT(m_iconsView->itemCategorizer() == 0);
675 m_iconsView->setItemCategorizer(new DolphinItemCategorizer());
676 } else {
677 KItemCategorizer* categorizer = m_iconsView->itemCategorizer();
678 m_iconsView->setItemCategorizer(0);
679 delete categorizer;
680 }
681 }
682 emit categorizedSortingChanged();
683 }
684
685 const DolphinView::Sorting sorting = props.sorting();
686 if (sorting != m_proxyModel->sorting()) {
687 m_proxyModel->setSorting(sorting);
688 emit sortingChanged(sorting);
689 }
690
691 const Qt::SortOrder sortOrder = props.sortOrder();
692 if (sortOrder != m_proxyModel->sortOrder()) {
693 m_proxyModel->setSortOrder(sortOrder);
694 emit sortOrderChanged(sortOrder);
695 }
696
697 KFileItemDelegate::AdditionalInformation info = props.additionalInfo();
698 if (info != m_fileItemDelegate->additionalInformation()) {
699 m_fileItemDelegate->setAdditionalInformation(info);
700
701 emit additionalInfoChanged(info);
702 }
703
704 const bool showPreview = props.showPreview();
705 if (showPreview != m_controller->showPreview()) {
706 m_controller->setShowPreview(showPreview);
707 emit showPreviewChanged();
708 }
709
710 startDirLister(url);
711 emit urlChanged(url);
712
713 m_statusBar->clear();
714 }
715
716 void DolphinView::triggerItem(const QModelIndex& index)
717 {
718 if (!isValidNameIndex(index)) {
719 return;
720 }
721
722 const Qt::KeyboardModifiers modifier = QApplication::keyboardModifiers();
723 if ((modifier & Qt::ShiftModifier) || (modifier & Qt::ControlModifier)) {
724 // items are selected by the user, hence don't trigger the
725 // item specified by 'index'
726 return;
727 }
728
729 KFileItem* item = m_dirModel->itemForIndex(m_proxyModel->mapToSource(index));
730 if (item == 0) {
731 return;
732 }
733
734 // Prefer the local path over the URL. This assures that the
735 // volume space information is correct. Assuming that the URL is media:/sda1,
736 // and the local path is /windows/C: For the URL the space info is related
737 // to the root partition (and hence wrong) and for the local path the space
738 // info is related to the windows partition (-> correct).
739 const QString localPath(item->localPath());
740 KUrl url;
741 if (localPath.isEmpty()) {
742 url = item->url();
743 } else {
744 url = localPath;
745 }
746
747 if (item->isDir()) {
748 setUrl(url);
749 } else if (item->isFile()) {
750 // allow to browse through ZIP and tar files
751 KMimeType::Ptr mime = item->mimeTypePtr();
752 if (mime->is("application/zip")) {
753 url.setProtocol("zip");
754 setUrl(url);
755 } else if (mime->is("application/x-tar") ||
756 mime->is("application/x-tarz") ||
757 mime->is("application/x-bzip-compressed-tar") ||
758 mime->is("application/x-compressed-tar") ||
759 mime->is("application/x-tzo")) {
760 url.setProtocol("tar");
761 setUrl(url);
762 } else {
763 item->run();
764 }
765 } else {
766 item->run();
767 }
768 }
769
770 void DolphinView::updateProgress(int percent)
771 {
772 if (m_showProgress) {
773 m_statusBar->setProgress(percent);
774 }
775 }
776
777 void DolphinView::updateItemCount()
778 {
779 if (m_showProgress) {
780 m_statusBar->setProgressText(QString());
781 m_statusBar->setProgress(100);
782 m_showProgress = false;
783 }
784
785 KFileItemList items(m_dirLister->items());
786 KFileItemList::const_iterator it = items.begin();
787 const KFileItemList::const_iterator end = items.end();
788
789 m_fileCount = 0;
790 m_folderCount = 0;
791
792 while (it != end) {
793 KFileItem* item = *it;
794 if (item->isDir()) {
795 ++m_folderCount;
796 } else {
797 ++m_fileCount;
798 }
799 ++it;
800 }
801
802 updateStatusBar();
803
804 m_blockContentsMovedSignal = false;
805 QTimer::singleShot(0, this, SLOT(restoreContentsPos()));
806 }
807
808 void DolphinView::generatePreviews(const KFileItemList& items)
809 {
810 if (m_controller->showPreview()) {
811
812 // Must turn QList<KFileItem *> to QList<KFileItem>...
813 QList<KFileItem> itemsToPreview;
814 foreach( KFileItem* it, items )
815 itemsToPreview.append( *it );
816
817 KIO::PreviewJob* job = KIO::filePreview(itemsToPreview, 128);
818 connect(job, SIGNAL(gotPreview(const KFileItem&, const QPixmap&)),
819 this, SLOT(showPreview(const KFileItem&, const QPixmap&)));
820 }
821 }
822
823 void DolphinView::showPreview(const KFileItem& item, const QPixmap& pixmap)
824 {
825 Q_ASSERT(!item.isNull());
826 if (item.url().directory() != m_dirLister->url().path()) {
827 // the preview job is still working on items of an older URL, hence
828 // the item is not part of the directory model anymore
829 return;
830 }
831
832 const QModelIndex idx = m_dirModel->indexForItem(item);
833 if (idx.isValid() && (idx.column() == 0)) {
834 const QMimeData* mimeData = QApplication::clipboard()->mimeData();
835 if (KonqMimeData::decodeIsCutSelection(mimeData) && isCutItem(item)) {
836 KIconEffect iconEffect;
837 const QPixmap cutPixmap = iconEffect.apply(pixmap, K3Icon::Desktop, K3Icon::DisabledState);
838 m_dirModel->setData(idx, QIcon(cutPixmap), Qt::DecorationRole);
839 } else {
840 m_dirModel->setData(idx, QIcon(pixmap), Qt::DecorationRole);
841 }
842 }
843 }
844
845 void DolphinView::restoreContentsPos()
846 {
847 KUrl currentUrl = m_urlNavigator->url();
848 if (!currentUrl.isEmpty()) {
849 QAbstractItemView* view = itemView();
850 // TODO: view->setCurrentItem(m_urlNavigator->currentFileName());
851 QPoint pos = m_urlNavigator->savedPosition();
852 view->horizontalScrollBar()->setValue(pos.x());
853 view->verticalScrollBar()->setValue(pos.y());
854 }
855 }
856
857 void DolphinView::showInfoMessage(const QString& msg)
858 {
859 m_statusBar->setMessage(msg, DolphinStatusBar::Information);
860 }
861
862 void DolphinView::showErrorMessage(const QString& msg)
863 {
864 m_statusBar->setMessage(msg, DolphinStatusBar::Error);
865 }
866
867 void DolphinView::emitSelectionChangedSignal()
868 {
869 emit selectionChanged(DolphinView::selectedItems());
870 }
871
872 void DolphinView::closeFilterBar()
873 {
874 m_filterBar->hide();
875 emit showFilterBarChanged(false);
876 }
877
878 void DolphinView::startDirLister(const KUrl& url, bool reload)
879 {
880 if (!url.isValid()) {
881 const QString location(url.pathOrUrl());
882 if (location.isEmpty()) {
883 m_statusBar->setMessage(i18n("The location is empty."), DolphinStatusBar::Error);
884 } else {
885 m_statusBar->setMessage(i18n("The location '%1' is invalid.", location),
886 DolphinStatusBar::Error);
887 }
888 return;
889 }
890
891 // Only show the directory loading progress if the status bar does
892 // not contain another progress information. This means that
893 // the directory loading progress information has the lowest priority.
894 const QString progressText(m_statusBar->progressText());
895 m_showProgress = progressText.isEmpty() ||
896 (progressText == i18n("Loading folder..."));
897 if (m_showProgress) {
898 m_statusBar->setProgressText(i18n("Loading folder..."));
899 m_statusBar->setProgress(0);
900 }
901
902 m_cutItemsCache.clear();
903 m_blockContentsMovedSignal = true;
904 m_dirLister->stop();
905
906 bool openDir = true;
907 bool keepOldDirs = isColumnViewActive() && !m_initializeColumnView;
908 m_initializeColumnView = false;
909
910 if (keepOldDirs) {
911 if (reload) {
912 keepOldDirs = false;
913
914 const KUrl& dirListerUrl = m_dirLister->url();
915 if (dirListerUrl.isValid()) {
916 const KUrl::List dirs = m_dirLister->directories();
917 KUrl url;
918 foreach(url, dirs) {
919 m_dirLister->updateDirectory(url);
920 }
921 openDir = false;
922 }
923 } else if (m_dirLister->directories().contains(url)) {
924 // The dir lister contains the directory already, so
925 // KDirLister::openUrl() may not been invoked twice.
926 m_dirLister->updateDirectory(url);
927 openDir = false;
928 } else {
929 const KUrl& dirListerUrl = m_dirLister->url();
930 if ((dirListerUrl == url) || !m_dirLister->url().isParentOf(url)) {
931 // The current URL is not a child of the dir lister
932 // URL. This may happen when e. g. a bookmark has been selected
933 // and hence the view must be reset.
934 keepOldDirs = false;
935 }
936 }
937 }
938
939 if (openDir) {
940 m_dirLister->openUrl(url, keepOldDirs, reload);
941 }
942 }
943
944 QString DolphinView::defaultStatusBarText() const
945 {
946 return KIO::itemsSummaryString(m_fileCount + m_folderCount,
947 m_fileCount,
948 m_folderCount,
949 0, false);
950 }
951
952 QString DolphinView::selectionStatusBarText() const
953 {
954 QString text;
955 const KFileItemList list = selectedItems();
956 if (list.isEmpty()) {
957 // when an item is triggered, it is temporary selected but selectedItems()
958 // will return an empty list
959 return QString();
960 }
961
962 int fileCount = 0;
963 int folderCount = 0;
964 KIO::filesize_t byteSize = 0;
965 KFileItemList::const_iterator it = list.begin();
966 const KFileItemList::const_iterator end = list.end();
967 while (it != end) {
968 KFileItem* item = *it;
969 if (item->isDir()) {
970 ++folderCount;
971 } else {
972 ++fileCount;
973 byteSize += item->size();
974 }
975 ++it;
976 }
977
978 if (folderCount > 0) {
979 text = i18np("1 Folder selected", "%1 Folders selected", folderCount);
980 if (fileCount > 0) {
981 text += ", ";
982 }
983 }
984
985 if (fileCount > 0) {
986 const QString sizeText(KIO::convertSize(byteSize));
987 text += i18np("1 File selected (%2)", "%1 Files selected (%2)", fileCount, sizeText);
988 }
989
990 return text;
991 }
992
993 void DolphinView::showFilterBar(bool show)
994 {
995 Q_ASSERT(m_filterBar != 0);
996 if (show) {
997 m_filterBar->show();
998 } else {
999 m_filterBar->hide();
1000 }
1001 }
1002
1003 void DolphinView::updateStatusBar()
1004 {
1005 // As the item count information is less important
1006 // in comparison with other messages, it should only
1007 // be shown if:
1008 // - the status bar is empty or
1009 // - shows already the item count information or
1010 // - shows only a not very important information
1011 // - if any progress is given don't show the item count info at all
1012 const QString msg(m_statusBar->message());
1013 const bool updateStatusBarMsg = (msg.isEmpty() ||
1014 (msg == m_statusBar->defaultText()) ||
1015 (m_statusBar->type() == DolphinStatusBar::Information)) &&
1016 (m_statusBar->progress() == 100);
1017
1018 const QString text(hasSelection() ? selectionStatusBarText() : defaultStatusBarText());
1019 m_statusBar->setDefaultText(text);
1020
1021 if (updateStatusBarMsg) {
1022 m_statusBar->setMessage(text, DolphinStatusBar::Default);
1023 }
1024 }
1025
1026 void DolphinView::requestActivation()
1027 {
1028 m_mainWindow->setActiveView(this);
1029 }
1030
1031 void DolphinView::changeSelection(const KFileItemList& selection)
1032 {
1033 clearSelection();
1034 if (selection.isEmpty()) {
1035 return;
1036 }
1037 KUrl baseUrl = url();
1038 KUrl url;
1039 QItemSelection new_selection;
1040 foreach(KFileItem* item, selection) {
1041 url = item->url().upUrl();
1042 if (baseUrl.equals(url, KUrl::CompareWithoutTrailingSlash)) {
1043 QModelIndex index = m_proxyModel->mapFromSource(m_dirModel->indexForItem(*item));
1044 new_selection.select(index, index);
1045 }
1046 }
1047 itemView()->selectionModel()->select(new_selection,
1048 QItemSelectionModel::ClearAndSelect
1049 | QItemSelectionModel::Current);
1050 }
1051
1052 void DolphinView::changeNameFilter(const QString& nameFilter)
1053 {
1054 // The name filter of KDirLister does a 'hard' filtering, which
1055 // means that only the items are shown where the names match
1056 // exactly the filter. This is non-transparent for the user, which
1057 // just wants to have a 'soft' filtering: does the name contain
1058 // the filter string?
1059 QString adjustedFilter(nameFilter);
1060 adjustedFilter.insert(0, '*');
1061 adjustedFilter.append('*');
1062
1063 // Use the ProxyModel to filter:
1064 // This code is #ifdefed as setNameFilter behaves
1065 // slightly different than the QSortFilterProxyModel
1066 // as it will not remove directories. I will ask
1067 // our beloved usability experts for input
1068 // -- z.
1069 #if 0
1070 m_dirLister->setNameFilter(adjustedFilter);
1071 m_dirLister->emitChanges();
1072 #else
1073 m_proxyModel->setFilterRegExp(nameFilter);
1074 #endif
1075 }
1076
1077 void DolphinView::openContextMenu(const QPoint& pos)
1078 {
1079 KFileItem* item = 0;
1080
1081 const QModelIndex index = itemView()->indexAt(pos);
1082 if (isValidNameIndex(index)) {
1083 item = fileItem(index);
1084 }
1085
1086 DolphinContextMenu contextMenu(m_mainWindow, item, url());
1087 contextMenu.open();
1088 }
1089
1090 void DolphinView::dropUrls(const KUrl::List& urls,
1091 const QModelIndex& index,
1092 QWidget* source)
1093 {
1094 KFileItem* directory = 0;
1095 if (isValidNameIndex(index)) {
1096 KFileItem* item = fileItem(index);
1097 Q_ASSERT(item != 0);
1098 if (item->isDir()) {
1099 // the URLs are dropped above a directory
1100 directory = item;
1101 }
1102 }
1103
1104 if ((directory == 0) && (source == itemView())) {
1105 // The dropping is done into the same viewport where
1106 // the dragging has been started. Just ignore this...
1107 return;
1108 }
1109
1110 const KUrl& destination = (directory == 0) ? url() :
1111 directory->url();
1112 dropUrls(urls, destination);
1113 }
1114
1115 void DolphinView::dropUrls(const KUrl::List& urls,
1116 const KUrl& destination)
1117 {
1118 m_mainWindow->dropUrls(urls, destination);
1119 }
1120
1121 void DolphinView::updateSorting(DolphinView::Sorting sorting)
1122 {
1123 ViewProperties props(url());
1124 props.setSorting(sorting);
1125
1126 m_proxyModel->setSorting(sorting);
1127
1128 emit sortingChanged(sorting);
1129 }
1130
1131 void DolphinView::updateSortOrder(Qt::SortOrder order)
1132 {
1133 ViewProperties props(url());
1134 props.setSortOrder(order);
1135
1136 m_proxyModel->setSortOrder(order);
1137
1138 emit sortOrderChanged(order);
1139 }
1140
1141 void DolphinView::emitContentsMoved()
1142 {
1143 if (!m_blockContentsMovedSignal) {
1144 emit contentsMoved(contentsX(), contentsY());
1145 }
1146 }
1147
1148 void DolphinView::updateActivationState()
1149 {
1150 m_urlNavigator->setActive(isActive());
1151 if (isActive()) {
1152 emit urlChanged(url());
1153 emit selectionChanged(selectedItems());
1154 }
1155 }
1156
1157 void DolphinView::updateCutItems()
1158 {
1159 // restore the icons of all previously selected items to the
1160 // original state...
1161 QList<CutItem>::const_iterator it = m_cutItemsCache.begin();
1162 QList<CutItem>::const_iterator end = m_cutItemsCache.end();
1163 while (it != end) {
1164 const QModelIndex index = m_dirModel->indexForUrl((*it).url);
1165 if (index.isValid()) {
1166 m_dirModel->setData(index, QIcon((*it).pixmap), Qt::DecorationRole);
1167 }
1168 ++it;
1169 }
1170 m_cutItemsCache.clear();
1171
1172 // ... and apply an item effect to all currently cut items
1173 applyCutItemEffect();
1174 }
1175
1176 void DolphinView::createView()
1177 {
1178 // delete current view
1179 QAbstractItemView* view = itemView();
1180 if (view != 0) {
1181 m_topLayout->removeWidget(view);
1182 view->close();
1183 if (view == m_iconsView) {
1184 KItemCategorizer* categorizer = m_iconsView->itemCategorizer();
1185 m_iconsView->setItemCategorizer(0);
1186 delete categorizer;
1187 }
1188 view->deleteLater();
1189 view = 0;
1190 m_iconsView = 0;
1191 m_detailsView = 0;
1192 m_columnView = 0;
1193 m_fileItemDelegate = 0;
1194 }
1195
1196 Q_ASSERT(m_iconsView == 0);
1197 Q_ASSERT(m_detailsView == 0);
1198 Q_ASSERT(m_columnView == 0);
1199
1200 // ... and recreate it representing the current mode
1201 switch (m_mode) {
1202 case IconsView:
1203 m_iconsView = new DolphinIconsView(this, m_controller);
1204 view = m_iconsView;
1205 break;
1206
1207 case DetailsView:
1208 m_detailsView = new DolphinDetailsView(this, m_controller);
1209 view = m_detailsView;
1210 break;
1211
1212 case ColumnView:
1213 m_columnView = new DolphinColumnView(this, m_controller);
1214 view = m_columnView;
1215 break;
1216 }
1217
1218 Q_ASSERT(view != 0);
1219
1220 m_fileItemDelegate = new KFileItemDelegate(view);
1221 view->setItemDelegate(m_fileItemDelegate);
1222
1223 view->setModel(m_proxyModel);
1224 view->setSelectionMode(QAbstractItemView::ExtendedSelection);
1225
1226 new KMimeTypeResolver(view, m_dirModel);
1227 m_topLayout->insertWidget(1, view);
1228
1229 connect(view->selectionModel(), SIGNAL(selectionChanged(const QItemSelection&, const QItemSelection&)),
1230 m_controller, SLOT(indicateSelectionChange()));
1231 connect(view->verticalScrollBar(), SIGNAL(valueChanged(int)),
1232 this, SLOT(emitContentsMoved()));
1233 connect(view->horizontalScrollBar(), SIGNAL(valueChanged(int)),
1234 this, SLOT(emitContentsMoved()));
1235 }
1236
1237 void DolphinView::selectAll(QItemSelectionModel::SelectionFlags flags)
1238 {
1239 QItemSelectionModel* selectionModel = itemView()->selectionModel();
1240 const QAbstractItemModel* itemModel = selectionModel->model();
1241
1242 const QModelIndex topLeft = itemModel->index(0, 0);
1243 const QModelIndex bottomRight = itemModel->index(itemModel->rowCount() - 1,
1244 itemModel->columnCount() - 1);
1245
1246 QItemSelection selection(topLeft, bottomRight);
1247 selectionModel->select(selection, flags);
1248 }
1249
1250 QAbstractItemView* DolphinView::itemView() const
1251 {
1252 if (m_detailsView != 0) {
1253 return m_detailsView;
1254 } else if (m_columnView != 0) {
1255 return m_columnView;
1256 }
1257
1258 return m_iconsView;
1259 }
1260
1261 bool DolphinView::isValidNameIndex(const QModelIndex& index) const
1262 {
1263 return index.isValid() && (index.column() == KDirModel::Name);
1264 }
1265
1266 bool DolphinView::isCutItem(const KFileItem& item) const
1267 {
1268 const QMimeData* mimeData = QApplication::clipboard()->mimeData();
1269 const KUrl::List cutUrls = KUrl::List::fromMimeData(mimeData);
1270
1271 const KUrl& itemUrl = item.url();
1272 KUrl::List::const_iterator it = cutUrls.begin();
1273 const KUrl::List::const_iterator end = cutUrls.end();
1274 while (it != end) {
1275 if (*it == itemUrl) {
1276 return true;
1277 }
1278 ++it;
1279 }
1280
1281 return false;
1282 }
1283
1284 void DolphinView::applyCutItemEffect()
1285 {
1286 const QMimeData* mimeData = QApplication::clipboard()->mimeData();
1287 if (!KonqMimeData::decodeIsCutSelection(mimeData)) {
1288 return;
1289 }
1290
1291 KFileItemList items(m_dirLister->items());
1292 KFileItemList::const_iterator it = items.begin();
1293 const KFileItemList::const_iterator end = items.end();
1294 while (it != end) {
1295 KFileItem* item = *it;
1296 if (isCutItem(*item)) {
1297 const QModelIndex index = m_dirModel->indexForItem(*item);
1298 const KFileItem* item = m_dirModel->itemForIndex(index);
1299 const QVariant value = m_dirModel->data(index, Qt::DecorationRole);
1300 if ((value.type() == QVariant::Icon) && (item != 0)) {
1301 const QIcon icon(qvariant_cast<QIcon>(value));
1302 QPixmap pixmap = icon.pixmap(128, 128);
1303
1304 // remember current pixmap for the item to be able
1305 // to restore it when other items get cut
1306 CutItem cutItem;
1307 cutItem.url = item->url();
1308 cutItem.pixmap = pixmap;
1309 m_cutItemsCache.append(cutItem);
1310
1311 // apply icon effect to the cut item
1312 KIconEffect iconEffect;
1313 pixmap = iconEffect.apply(pixmap, K3Icon::Desktop, K3Icon::DisabledState);
1314 m_dirModel->setData(index, QIcon(pixmap), Qt::DecorationRole);
1315 }
1316 }
1317 ++it;
1318 }
1319 }
1320
1321 #include "dolphinview.moc"