]> cloud.milkyroute.net Git - dolphin.git/blob - src/dolphinmainwindow.cpp
Make Escape move focus from location bar to view
[dolphin.git] / src / dolphinmainwindow.cpp
1 /*
2 * SPDX-FileCopyrightText: 2006 Peter Penz <peter.penz19@gmail.com>
3 * SPDX-FileCopyrightText: 2006 Stefan Monov <logixoul@gmail.com>
4 * SPDX-FileCopyrightText: 2006 Cvetoslav Ludmiloff <ludmiloff@gmail.com>
5 *
6 * SPDX-License-Identifier: GPL-2.0-or-later
7 */
8
9 #include "dolphinmainwindow.h"
10
11 #include "admin/workerintegration.h"
12 #include "dolphin_generalsettings.h"
13 #include "dolphinbookmarkhandler.h"
14 #include "dolphincontextmenu.h"
15 #include "dolphindockwidget.h"
16 #include "dolphinmainwindowadaptor.h"
17 #include "dolphinnavigatorswidgetaction.h"
18 #include "dolphinnewfilemenu.h"
19 #include "dolphinplacesmodelsingleton.h"
20 #include "dolphinrecenttabsmenu.h"
21 #include "dolphintabpage.h"
22 #include "dolphinurlnavigatorscontroller.h"
23 #include "dolphinviewcontainer.h"
24 #include "global.h"
25 #include "middleclickactioneventfilter.h"
26 #include "panels/folders/folderspanel.h"
27 #include "panels/places/placespanel.h"
28 #include "panels/terminal/terminalpanel.h"
29 #include "selectionmode/actiontexthelper.h"
30 #include "settings/dolphinsettingsdialog.h"
31 #include "statusbar/dolphinstatusbar.h"
32 #include "views/dolphinnewfilemenuobserver.h"
33 #include "views/dolphinremoteencoding.h"
34 #include "views/dolphinviewactionhandler.h"
35 #include "views/draganddrophelper.h"
36 #include "views/viewproperties.h"
37
38 #include <KActionCollection>
39 #include <KActionMenu>
40 #include <KAuthorized>
41 #include <KColorSchemeManager>
42 #include <KConfig>
43 #include <KConfigGui>
44 #include <KDialogJobUiDelegate>
45 #include <KDualAction>
46 #include <KFileItemListProperties>
47 #include <KIO/CommandLauncherJob>
48 #include <KIO/JobUiDelegateFactory>
49 #include <KIO/OpenFileManagerWindowJob>
50 #include <KIO/OpenUrlJob>
51 #include <KJobWidgets>
52 #include <KLocalizedString>
53 #include <KMessageBox>
54 #include <KProtocolInfo>
55 #include <KProtocolManager>
56 #include <KRecentFilesAction>
57 #include <KRuntimePlatform>
58 #include <KShell>
59 #include <KShortcutsDialog>
60 #include <KStandardAction>
61 #include <KSycoca>
62 #include <KTerminalLauncherJob>
63 #include <KToggleAction>
64 #include <KToolBar>
65 #include <KToolBarPopupAction>
66 #include <KUrlComboBox>
67 #include <KUrlNavigator>
68 #include <KWindowSystem>
69 #include <KXMLGUIFactory>
70
71 #include <kwidgetsaddons_version.h>
72
73 #include <QApplication>
74 #include <QClipboard>
75 #include <QCloseEvent>
76 #include <QDesktopServices>
77 #include <QDialog>
78 #include <QDomDocument>
79 #include <QFileInfo>
80 #include <QLineEdit>
81 #include <QMenuBar>
82 #include <QPushButton>
83 #include <QShowEvent>
84 #include <QStandardPaths>
85 #include <QTimer>
86 #include <QToolButton>
87 #include <QtConcurrentRun>
88 #include <dolphindebug.h>
89
90 #include <algorithm>
91
92 #if HAVE_X11
93 #include <KStartupInfo>
94 #endif
95
96 namespace
97 {
98 // Used for GeneralSettings::version() to determine whether
99 // an updated version of Dolphin is running, so as to migrate
100 // removed/renamed ...etc config entries; increment it in such
101 // cases
102 const int CurrentDolphinVersion = 202;
103 // The maximum number of entries in the back/forward popup menu
104 const int MaxNumberOfNavigationentries = 12;
105 // The maximum number of "Go to Tab" shortcuts
106 const int MaxActivateTabShortcuts = 9;
107 }
108
109 DolphinMainWindow::DolphinMainWindow()
110 : KXmlGuiWindow(nullptr)
111 , m_newFileMenu(nullptr)
112 , m_tabWidget(nullptr)
113 , m_activeViewContainer(nullptr)
114 , m_actionHandler(nullptr)
115 , m_remoteEncoding(nullptr)
116 , m_settingsDialog()
117 , m_bookmarkHandler(nullptr)
118 , m_disabledActionNotifier(nullptr)
119 , m_lastHandleUrlOpenJob(nullptr)
120 , m_terminalPanel(nullptr)
121 , m_placesPanel(nullptr)
122 , m_tearDownFromPlacesRequested(false)
123 , m_backAction(nullptr)
124 , m_forwardAction(nullptr)
125 , m_splitViewAction(nullptr)
126 , m_splitViewMenuAction(nullptr)
127 , m_sessionSaveTimer(nullptr)
128 , m_sessionSaveWatcher(nullptr)
129 , m_sessionSaveScheduled(false)
130 {
131 Q_INIT_RESOURCE(dolphin);
132
133 new MainWindowAdaptor(this);
134
135 #ifndef Q_OS_WIN
136 setWindowFlags(Qt::WindowContextHelpButtonHint);
137 #endif
138 setComponentName(QStringLiteral("dolphin"), QGuiApplication::applicationDisplayName());
139 setObjectName(QStringLiteral("Dolphin#"));
140
141 setStateConfigGroup("State");
142
143 #if defined(Q_OS_WIN) || defined(Q_OS_MACOS)
144 new KColorSchemeManager(this); // Sets a sensible color scheme which fixes unreadable icons and text on Windows.
145 #endif
146
147 connect(&DolphinNewFileMenuObserver::instance(), &DolphinNewFileMenuObserver::errorMessage, this, &DolphinMainWindow::showErrorMessage);
148
149 KIO::FileUndoManager *undoManager = KIO::FileUndoManager::self();
150 undoManager->setUiInterface(new UndoUiInterface());
151
152 connect(undoManager, &KIO::FileUndoManager::undoAvailable, this, &DolphinMainWindow::slotUndoAvailable);
153 connect(undoManager, &KIO::FileUndoManager::undoTextChanged, this, &DolphinMainWindow::slotUndoTextChanged);
154 connect(undoManager, &KIO::FileUndoManager::jobRecordingStarted, this, &DolphinMainWindow::clearStatusBar);
155 connect(undoManager, &KIO::FileUndoManager::jobRecordingFinished, this, &DolphinMainWindow::showCommand);
156
157 const bool firstRun = (GeneralSettings::version() < 200);
158 if (firstRun) {
159 GeneralSettings::setViewPropsTimestamp(QDateTime::currentDateTime());
160 }
161
162 setAcceptDrops(true);
163
164 auto *navigatorsWidgetAction = new DolphinNavigatorsWidgetAction(this);
165 actionCollection()->addAction(QStringLiteral("url_navigators"), navigatorsWidgetAction);
166 m_tabWidget = new DolphinTabWidget(navigatorsWidgetAction, this);
167 m_tabWidget->setObjectName("tabWidget");
168 connect(m_tabWidget, &DolphinTabWidget::activeViewChanged, this, &DolphinMainWindow::activeViewChanged);
169 connect(m_tabWidget, &DolphinTabWidget::tabCountChanged, this, &DolphinMainWindow::tabCountChanged);
170 connect(m_tabWidget, &DolphinTabWidget::currentUrlChanged, this, &DolphinMainWindow::updateWindowTitle);
171 setCentralWidget(m_tabWidget);
172
173 m_actionTextHelper = new SelectionMode::ActionTextHelper(this);
174 setupActions();
175
176 m_actionHandler = new DolphinViewActionHandler(actionCollection(), m_actionTextHelper, this);
177 connect(m_actionHandler, &DolphinViewActionHandler::actionBeingHandled, this, &DolphinMainWindow::clearStatusBar);
178 connect(m_actionHandler, &DolphinViewActionHandler::createDirectoryTriggered, this, &DolphinMainWindow::createDirectory);
179 connect(m_actionHandler, &DolphinViewActionHandler::selectionModeChangeTriggered, this, &DolphinMainWindow::slotSetSelectionMode);
180
181 Q_CHECK_PTR(actionCollection()->action(QStringLiteral("create_dir")));
182 m_newFileMenu->setNewFolderShortcutAction(actionCollection()->action(QStringLiteral("create_dir")));
183
184 m_remoteEncoding = new DolphinRemoteEncoding(this, m_actionHandler);
185 connect(this, &DolphinMainWindow::urlChanged, m_remoteEncoding, &DolphinRemoteEncoding::slotAboutToOpenUrl);
186
187 m_disabledActionNotifier = new DisabledActionNotifier(this);
188 connect(m_disabledActionNotifier, &DisabledActionNotifier::disabledActionTriggered, this, [this](const QAction *, QString reason) {
189 m_activeViewContainer->showMessage(reason, KMessageWidget::Warning);
190 });
191
192 setupDockWidgets();
193
194 const bool usePhoneUi{KRuntimePlatform::runtimePlatform().contains(QLatin1String("phone"))};
195 setupGUI(Save | Create | ToolBar, usePhoneUi ? QStringLiteral("dolphinuiforphones.rc") : QString() /* load the default dolphinui.rc file */);
196 stateChanged(QStringLiteral("new_file"));
197
198 QClipboard *clipboard = QApplication::clipboard();
199 connect(clipboard, &QClipboard::dataChanged, this, &DolphinMainWindow::updatePasteAction);
200
201 QAction *toggleFilterBarAction = actionCollection()->action(QStringLiteral("toggle_filter"));
202 toggleFilterBarAction->setChecked(GeneralSettings::filterBar());
203
204 if (firstRun) {
205 menuBar()->setVisible(false);
206
207 if (usePhoneUi) {
208 Q_ASSERT(qobject_cast<QDockWidget *>(m_placesPanel->parent()));
209 m_placesPanel->parentWidget()->hide();
210 auto settings = GeneralSettings::self();
211 settings->setShowZoomSlider(false); // Zooming can be done with pinch gestures instead and we are short on horizontal space.
212 settings->setRenameInline(false); // This works around inline renaming currently not working well with virtual keyboards.
213 settings->save(); // Otherwise the RenameInline setting is not picked up for the first time Dolphin is used.
214 }
215 }
216
217 const bool showMenu = !menuBar()->isHidden();
218 QAction *showMenuBarAction = actionCollection()->action(KStandardAction::name(KStandardAction::ShowMenubar));
219 showMenuBarAction->setChecked(showMenu); // workaround for bug #171080
220
221 auto hamburgerMenu = static_cast<KHamburgerMenu *>(actionCollection()->action(KStandardAction::name(KStandardAction::HamburgerMenu)));
222 hamburgerMenu->setMenuBar(menuBar());
223 hamburgerMenu->setShowMenuBarAction(showMenuBarAction);
224 connect(hamburgerMenu, &KHamburgerMenu::aboutToShowMenu, this, &DolphinMainWindow::updateHamburgerMenu);
225 hamburgerMenu->hideActionsOf(toolBar());
226 if (GeneralSettings::version() < 201 && !toolBar()->actions().contains(hamburgerMenu)) {
227 addHamburgerMenuToToolbar();
228 }
229
230 updateAllowedToolbarAreas();
231
232 // enable middle-click on back/forward/up to open in a new tab
233 auto *middleClickEventFilter = new MiddleClickActionEventFilter(this);
234 connect(middleClickEventFilter, &MiddleClickActionEventFilter::actionMiddleClicked, this, &DolphinMainWindow::slotToolBarActionMiddleClicked);
235 toolBar()->installEventFilter(middleClickEventFilter);
236
237 setupWhatsThis();
238
239 connect(KSycoca::self(), &KSycoca::databaseChanged, this, &DolphinMainWindow::updateOpenPreferredSearchToolAction);
240
241 QTimer::singleShot(0, this, &DolphinMainWindow::updateOpenPreferredSearchToolAction);
242
243 m_fileItemActions.setParentWidget(this);
244 connect(&m_fileItemActions, &KFileItemActions::error, this, [this](const QString &errorMessage) {
245 showErrorMessage(errorMessage);
246 });
247
248 connect(GeneralSettings::self(), &GeneralSettings::splitViewChanged, this, &DolphinMainWindow::slotSplitViewChanged);
249 }
250
251 DolphinMainWindow::~DolphinMainWindow()
252 {
253 // This fixes a crash on Wayland when closing the mainwindow while another dialog is open.
254 disconnect(QGuiApplication::clipboard(), &QClipboard::dataChanged, this, &DolphinMainWindow::updatePasteAction);
255
256 // This fixes a crash in dolphinmainwindowtest where the connection below fires even though the KMainWindow destructor of this object is already running.
257 Q_CHECK_PTR(qobject_cast<DolphinDockWidget *>(m_placesPanel->parent()));
258 disconnect(static_cast<DolphinDockWidget *>(m_placesPanel->parent()),
259 &DolphinDockWidget::visibilityChanged,
260 this,
261 &DolphinMainWindow::slotPlacesPanelVisibilityChanged);
262 }
263
264 QVector<DolphinViewContainer *> DolphinMainWindow::viewContainers() const
265 {
266 QVector<DolphinViewContainer *> viewContainers;
267
268 for (int i = 0; i < m_tabWidget->count(); ++i) {
269 DolphinTabPage *tabPage = m_tabWidget->tabPageAt(i);
270
271 viewContainers << tabPage->primaryViewContainer();
272 if (tabPage->splitViewEnabled()) {
273 viewContainers << tabPage->secondaryViewContainer();
274 }
275 }
276 return viewContainers;
277 }
278
279 void DolphinMainWindow::openDirectories(const QList<QUrl> &dirs, bool splitView)
280 {
281 m_tabWidget->openDirectories(dirs, splitView);
282 }
283
284 void DolphinMainWindow::openDirectories(const QStringList &dirs, bool splitView)
285 {
286 openDirectories(QUrl::fromStringList(dirs), splitView);
287 }
288
289 void DolphinMainWindow::openFiles(const QList<QUrl> &files, bool splitView)
290 {
291 m_tabWidget->openFiles(files, splitView);
292 }
293
294 bool DolphinMainWindow::isFoldersPanelEnabled() const
295 {
296 return actionCollection()->action(QStringLiteral("show_folders_panel"))->isChecked();
297 }
298
299 bool DolphinMainWindow::isInformationPanelEnabled() const
300 {
301 #if HAVE_BALOO
302 return actionCollection()->action(QStringLiteral("show_information_panel"))->isChecked();
303 #else
304 return false;
305 #endif
306 }
307
308 bool DolphinMainWindow::isSplitViewEnabledInCurrentTab() const
309 {
310 return m_tabWidget->currentTabPage()->splitViewEnabled();
311 }
312
313 void DolphinMainWindow::openFiles(const QStringList &files, bool splitView)
314 {
315 openFiles(QUrl::fromStringList(files), splitView);
316 }
317
318 void DolphinMainWindow::activateWindow(const QString &activationToken)
319 {
320 window()->setAttribute(Qt::WA_NativeWindow, true);
321
322 if (KWindowSystem::isPlatformWayland()) {
323 KWindowSystem::setCurrentXdgActivationToken(activationToken);
324 } else if (KWindowSystem::isPlatformX11()) {
325 #if HAVE_X11
326 KStartupInfo::setNewStartupId(window()->windowHandle(), activationToken.toUtf8());
327 #endif
328 }
329
330 KWindowSystem::activateWindow(window()->windowHandle());
331 }
332
333 bool DolphinMainWindow::isActiveWindow()
334 {
335 return window()->isActiveWindow();
336 }
337
338 void DolphinMainWindow::showCommand(CommandType command)
339 {
340 DolphinStatusBar *statusBar = m_activeViewContainer->statusBar();
341 switch (command) {
342 case KIO::FileUndoManager::Copy:
343 statusBar->setText(i18nc("@info:status", "Successfully copied."));
344 break;
345 case KIO::FileUndoManager::Move:
346 statusBar->setText(i18nc("@info:status", "Successfully moved."));
347 break;
348 case KIO::FileUndoManager::Link:
349 statusBar->setText(i18nc("@info:status", "Successfully linked."));
350 break;
351 case KIO::FileUndoManager::Trash:
352 statusBar->setText(i18nc("@info:status", "Successfully moved to trash."));
353 break;
354 case KIO::FileUndoManager::Rename:
355 statusBar->setText(i18nc("@info:status", "Successfully renamed."));
356 break;
357
358 case KIO::FileUndoManager::Mkdir:
359 statusBar->setText(i18nc("@info:status", "Created folder."));
360 break;
361
362 default:
363 break;
364 }
365 }
366
367 void DolphinMainWindow::pasteIntoFolder()
368 {
369 m_activeViewContainer->view()->pasteIntoFolder();
370 }
371
372 void DolphinMainWindow::changeUrl(const QUrl &url)
373 {
374 if (!KProtocolManager::supportsListing(url)) {
375 // The URL navigator only checks for validity, not
376 // if the URL can be listed. An error message is
377 // shown due to DolphinViewContainer::restoreView().
378 return;
379 }
380
381 m_activeViewContainer->setUrl(url);
382 updateFileAndEditActions();
383 updatePasteAction();
384 updateViewActions();
385 updateGoActions();
386
387 // will signal used urls to activities manager, too
388 m_recentFiles->addUrl(url);
389
390 Q_EMIT urlChanged(url);
391 }
392
393 void DolphinMainWindow::slotTerminalDirectoryChanged(const QUrl &url)
394 {
395 if (m_tearDownFromPlacesRequested && url == QUrl::fromLocalFile(QDir::homePath())) {
396 m_placesPanel->proceedWithTearDown();
397 m_tearDownFromPlacesRequested = false;
398 }
399
400 m_activeViewContainer->setAutoGrabFocus(false);
401 changeUrl(url);
402 m_activeViewContainer->setAutoGrabFocus(true);
403 }
404
405 void DolphinMainWindow::slotEditableStateChanged(bool editable)
406 {
407 KToggleAction *editableLocationAction = static_cast<KToggleAction *>(actionCollection()->action(QStringLiteral("editable_location")));
408 editableLocationAction->setChecked(editable);
409 }
410
411 void DolphinMainWindow::slotSelectionChanged(const KFileItemList &selection)
412 {
413 updateFileAndEditActions();
414
415 const int selectedUrlsCount = m_tabWidget->currentTabPage()->selectedItemsCount();
416
417 QAction *compareFilesAction = actionCollection()->action(QStringLiteral("compare_files"));
418 if (selectedUrlsCount == 2) {
419 compareFilesAction->setEnabled(isKompareInstalled());
420 } else {
421 compareFilesAction->setEnabled(false);
422 }
423
424 Q_EMIT selectionChanged(selection);
425 }
426
427 void DolphinMainWindow::updateHistory()
428 {
429 const KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
430 const int index = urlNavigator->historyIndex();
431
432 QAction *backAction = actionCollection()->action(KStandardAction::name(KStandardAction::Back));
433 if (backAction) {
434 backAction->setToolTip(i18nc("@info", "Go back"));
435 backAction->setWhatsThis(i18nc("@info:whatsthis go back", "Return to the previously viewed folder."));
436 backAction->setEnabled(index < urlNavigator->historySize() - 1);
437 }
438
439 QAction *forwardAction = actionCollection()->action(KStandardAction::name(KStandardAction::Forward));
440 if (forwardAction) {
441 forwardAction->setToolTip(i18nc("@info", "Go forward"));
442 forwardAction->setWhatsThis(xi18nc("@info:whatsthis go forward", "This undoes a <interface>Go|Back</interface> action."));
443 forwardAction->setEnabled(index > 0);
444 }
445 }
446
447 void DolphinMainWindow::updateFilterBarAction(bool show)
448 {
449 QAction *toggleFilterBarAction = actionCollection()->action(QStringLiteral("toggle_filter"));
450 toggleFilterBarAction->setChecked(show);
451 }
452
453 void DolphinMainWindow::openNewMainWindow()
454 {
455 Dolphin::openNewWindow({m_activeViewContainer->url()}, this);
456 }
457
458 void DolphinMainWindow::openNewActivatedTab()
459 {
460 // keep browsers compatibility, new tab is always after last one
461 auto openNewTabAfterLastTabConfigured = GeneralSettings::openNewTabAfterLastTab();
462 GeneralSettings::setOpenNewTabAfterLastTab(true);
463 m_tabWidget->openNewActivatedTab();
464 GeneralSettings::setOpenNewTabAfterLastTab(openNewTabAfterLastTabConfigured);
465 }
466
467 void DolphinMainWindow::addToPlaces()
468 {
469 QUrl url;
470 QString name;
471
472 // If nothing is selected, act on the current dir
473 if (m_activeViewContainer->view()->selectedItems().isEmpty()) {
474 url = m_activeViewContainer->url();
475 name = m_activeViewContainer->placesText();
476 } else {
477 const auto dirToAdd = m_activeViewContainer->view()->selectedItems().first();
478 url = dirToAdd.url();
479 name = dirToAdd.name();
480 }
481 if (url.isValid()) {
482 QString icon;
483 if (m_activeViewContainer->isSearchModeEnabled()) {
484 icon = QStringLiteral("folder-saved-search-symbolic");
485 } else {
486 icon = KIO::iconNameForUrl(url);
487 }
488 DolphinPlacesModelSingleton::instance().placesModel()->addPlace(name, url, icon);
489 }
490 }
491
492 void DolphinMainWindow::openNewTab(const QUrl &url)
493 {
494 m_tabWidget->openNewTab(url, QUrl());
495 }
496
497 void DolphinMainWindow::openNewTabAndActivate(const QUrl &url)
498 {
499 m_tabWidget->openNewActivatedTab(url, QUrl());
500 }
501
502 void DolphinMainWindow::openNewWindow(const QUrl &url)
503 {
504 Dolphin::openNewWindow({url}, this);
505 }
506
507 void DolphinMainWindow::slotSplitViewChanged()
508 {
509 m_tabWidget->currentTabPage()->setSplitViewEnabled(GeneralSettings::splitView(), WithAnimation);
510 updateSplitActions();
511 }
512
513 void DolphinMainWindow::openInNewTab()
514 {
515 const KFileItemList &list = m_activeViewContainer->view()->selectedItems();
516 bool tabCreated = false;
517
518 for (const KFileItem &item : list) {
519 const QUrl &url = DolphinView::openItemAsFolderUrl(item);
520 if (!url.isEmpty()) {
521 openNewTab(url);
522 tabCreated = true;
523 }
524 }
525
526 // if no new tab has been created from the selection
527 // open the current directory in a new tab
528 if (!tabCreated) {
529 openNewTab(m_activeViewContainer->url());
530 }
531 }
532
533 void DolphinMainWindow::openInNewWindow()
534 {
535 QUrl newWindowUrl;
536
537 const KFileItemList list = m_activeViewContainer->view()->selectedItems();
538 if (list.isEmpty()) {
539 newWindowUrl = m_activeViewContainer->url();
540 } else if (list.count() == 1) {
541 const KFileItem &item = list.first();
542 newWindowUrl = DolphinView::openItemAsFolderUrl(item);
543 }
544
545 if (!newWindowUrl.isEmpty()) {
546 Dolphin::openNewWindow({newWindowUrl}, this);
547 }
548 }
549
550 void DolphinMainWindow::openInSplitView(const QUrl &url)
551 {
552 QUrl newSplitViewUrl = url;
553
554 if (newSplitViewUrl.isEmpty()) {
555 const KFileItemList list = m_activeViewContainer->view()->selectedItems();
556 if (list.count() == 1) {
557 const KFileItem &item = list.first();
558 newSplitViewUrl = DolphinView::openItemAsFolderUrl(item);
559 }
560 }
561
562 if (newSplitViewUrl.isEmpty()) {
563 return;
564 }
565
566 DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
567 if (tabPage->splitViewEnabled()) {
568 tabPage->switchActiveView();
569 tabPage->activeViewContainer()->setUrl(newSplitViewUrl);
570 } else {
571 tabPage->setSplitViewEnabled(true, WithAnimation, newSplitViewUrl);
572 updateViewActions();
573 }
574 }
575
576 void DolphinMainWindow::showTarget()
577 {
578 const KFileItem link = m_activeViewContainer->view()->selectedItems().at(0);
579 const QUrl destinationUrl = link.url().resolved(QUrl(link.linkDest()));
580
581 auto job = KIO::stat(destinationUrl, KIO::StatJob::SourceSide, KIO::StatNoDetails);
582
583 connect(job, &KJob::finished, this, [this, destinationUrl](KJob *job) {
584 KIO::StatJob *statJob = static_cast<KIO::StatJob *>(job);
585
586 if (statJob->error()) {
587 m_activeViewContainer->showMessage(job->errorString(), KMessageWidget::Error);
588 } else {
589 KIO::highlightInFileManager({destinationUrl});
590 }
591 });
592 }
593
594 bool DolphinMainWindow::event(QEvent *event)
595 {
596 if (event->type() == QEvent::ShortcutOverride) {
597 const QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
598 if (keyEvent->key() == Qt::Key_Space && m_activeViewContainer->view()->handleSpaceAsNormalKey()) {
599 event->accept();
600 return true;
601 }
602 }
603
604 return KXmlGuiWindow::event(event);
605 }
606
607 void DolphinMainWindow::showEvent(QShowEvent *event)
608 {
609 KXmlGuiWindow::showEvent(event);
610
611 if (!event->spontaneous() && m_activeViewContainer) {
612 m_activeViewContainer->view()->setFocus();
613 }
614 }
615
616 void DolphinMainWindow::closeEvent(QCloseEvent *event)
617 {
618 // Find out if Dolphin is closed directly by the user or
619 // by the session manager because the session is closed
620 bool closedByUser = true;
621 if (qApp->isSavingSession()) {
622 closedByUser = false;
623 }
624
625 if (m_tabWidget->count() > 1 && GeneralSettings::confirmClosingMultipleTabs() && !GeneralSettings::rememberOpenedTabs() && closedByUser) {
626 // Ask the user if he really wants to quit and close all tabs.
627 // Open a confirmation dialog with 3 buttons:
628 // QDialogButtonBox::Yes -> Quit
629 // QDialogButtonBox::No -> Close only the current tab
630 // QDialogButtonBox::Cancel -> do nothing
631 QDialog *dialog = new QDialog(this, Qt::Dialog);
632 dialog->setWindowTitle(i18nc("@title:window", "Confirmation"));
633 dialog->setModal(true);
634 QDialogButtonBox *buttons = new QDialogButtonBox(QDialogButtonBox::Yes | QDialogButtonBox::No | QDialogButtonBox::Cancel);
635 KGuiItem::assign(buttons->button(QDialogButtonBox::Yes),
636 KGuiItem(i18nc("@action:button 'Quit Dolphin' button", "&Quit %1", QGuiApplication::applicationDisplayName()),
637 QIcon::fromTheme(QStringLiteral("application-exit"))));
638 KGuiItem::assign(buttons->button(QDialogButtonBox::No), KGuiItem(i18n("C&lose Current Tab"), QIcon::fromTheme(QStringLiteral("tab-close"))));
639 KGuiItem::assign(buttons->button(QDialogButtonBox::Cancel), KStandardGuiItem::cancel());
640 buttons->button(QDialogButtonBox::Yes)->setDefault(true);
641
642 bool doNotAskAgainCheckboxResult = false;
643
644 const auto result = KMessageBox::createKMessageBox(dialog,
645 buttons,
646 QMessageBox::Warning,
647 i18n("You have multiple tabs open in this window, are you sure you want to quit?"),
648 QStringList(),
649 i18n("Do not ask again"),
650 &doNotAskAgainCheckboxResult,
651 KMessageBox::Notify);
652
653 if (doNotAskAgainCheckboxResult) {
654 GeneralSettings::setConfirmClosingMultipleTabs(false);
655 }
656
657 switch (result) {
658 case QDialogButtonBox::Yes:
659 // Quit
660 break;
661 case QDialogButtonBox::No:
662 // Close only the current tab
663 m_tabWidget->closeTab();
664 Q_FALLTHROUGH();
665 default:
666 event->ignore();
667 return;
668 }
669 }
670
671 if (m_terminalPanel && m_terminalPanel->hasProgramRunning() && GeneralSettings::confirmClosingTerminalRunningProgram() && closedByUser) {
672 // Ask if the user really wants to quit Dolphin with a program that is still running in the Terminal panel
673 // Open a confirmation dialog with 3 buttons:
674 // QDialogButtonBox::Yes -> Quit
675 // QDialogButtonBox::No -> Show Terminal Panel
676 // QDialogButtonBox::Cancel -> do nothing
677 QDialog *dialog = new QDialog(this, Qt::Dialog);
678 dialog->setWindowTitle(i18nc("@title:window", "Confirmation"));
679 dialog->setModal(true);
680 auto standardButtons = QDialogButtonBox::Yes | QDialogButtonBox::Cancel;
681 if (!m_terminalPanel->isVisible()) {
682 standardButtons |= QDialogButtonBox::No;
683 }
684 QDialogButtonBox *buttons = new QDialogButtonBox(standardButtons);
685 KGuiItem::assign(buttons->button(QDialogButtonBox::Yes), KStandardGuiItem::quit());
686 if (!m_terminalPanel->isVisible()) {
687 KGuiItem::assign(buttons->button(QDialogButtonBox::No), KGuiItem(i18n("Show &Terminal Panel"), QIcon::fromTheme(QStringLiteral("dialog-scripts"))));
688 }
689 KGuiItem::assign(buttons->button(QDialogButtonBox::Cancel), KStandardGuiItem::cancel());
690
691 bool doNotAskAgainCheckboxResult = false;
692
693 const auto result = KMessageBox::createKMessageBox(
694 dialog,
695 buttons,
696 QMessageBox::Warning,
697 i18n("The program '%1' is still running in the Terminal panel. Are you sure you want to quit?", m_terminalPanel->runningProgramName()),
698 QStringList(),
699 i18n("Do not ask again"),
700 &doNotAskAgainCheckboxResult,
701 KMessageBox::Notify | KMessageBox::Dangerous);
702
703 if (doNotAskAgainCheckboxResult) {
704 GeneralSettings::setConfirmClosingTerminalRunningProgram(false);
705 }
706
707 switch (result) {
708 case QDialogButtonBox::Yes:
709 // Quit
710 break;
711 case QDialogButtonBox::No:
712 actionCollection()->action("show_terminal_panel")->trigger();
713 // Do not quit, ignore quit event
714 Q_FALLTHROUGH();
715 default:
716 event->ignore();
717 return;
718 }
719 }
720
721 if (m_sessionSaveTimer && (m_sessionSaveTimer->isActive() || m_sessionSaveWatcher->isRunning())) {
722 const bool sessionSaveTimerActive = m_sessionSaveTimer->isActive();
723
724 m_sessionSaveTimer->stop();
725 m_sessionSaveWatcher->disconnect();
726 m_sessionSaveWatcher->waitForFinished();
727
728 if (sessionSaveTimerActive || m_sessionSaveScheduled) {
729 slotSaveSession();
730 }
731 }
732
733 GeneralSettings::setVersion(CurrentDolphinVersion);
734 GeneralSettings::self()->save();
735
736 KXmlGuiWindow::closeEvent(event);
737 }
738
739 void DolphinMainWindow::slotSaveSession()
740 {
741 m_sessionSaveScheduled = false;
742
743 if (m_sessionSaveWatcher->isRunning()) {
744 // The previous session is still being saved - schedule another save.
745 m_sessionSaveWatcher->disconnect();
746 connect(m_sessionSaveWatcher, &QFutureWatcher<void>::finished, this, &DolphinMainWindow::slotSaveSession, Qt::SingleShotConnection);
747 m_sessionSaveScheduled = true;
748 } else if (!m_sessionSaveTimer->isActive()) {
749 // No point in saving the session if the timer is running (since it will save the session again when it times out).
750 KConfigGui::setSessionConfig(QStringLiteral("dolphin"), QStringLiteral("dolphin"));
751 KConfig *config = KConfigGui::sessionConfig();
752 saveGlobalProperties(config);
753 savePropertiesInternal(config, 1);
754
755 auto future = QtConcurrent::run([config]() {
756 config->sync();
757 });
758 m_sessionSaveWatcher->setFuture(future);
759 }
760 }
761
762 void DolphinMainWindow::setSessionAutoSaveEnabled(bool enable)
763 {
764 if (enable) {
765 if (!m_sessionSaveTimer) {
766 m_sessionSaveTimer = new QTimer(this);
767 m_sessionSaveWatcher = new QFutureWatcher<void>(this);
768 m_sessionSaveTimer->setSingleShot(true);
769 m_sessionSaveTimer->setInterval(22000);
770
771 connect(m_sessionSaveTimer, &QTimer::timeout, this, &DolphinMainWindow::slotSaveSession);
772 }
773
774 connect(m_tabWidget, &DolphinTabWidget::urlChanged, m_sessionSaveTimer, qOverload<>(&QTimer::start), Qt::UniqueConnection);
775 connect(m_tabWidget, &DolphinTabWidget::tabCountChanged, m_sessionSaveTimer, qOverload<>(&QTimer::start), Qt::UniqueConnection);
776 connect(m_tabWidget, &DolphinTabWidget::activeViewChanged, m_sessionSaveTimer, qOverload<>(&QTimer::start), Qt::UniqueConnection);
777 } else if (m_sessionSaveTimer) {
778 m_sessionSaveTimer->stop();
779 m_sessionSaveWatcher->disconnect();
780 m_sessionSaveScheduled = false;
781
782 m_sessionSaveWatcher->waitForFinished();
783
784 m_sessionSaveTimer->deleteLater();
785 m_sessionSaveWatcher->deleteLater();
786 m_sessionSaveTimer = nullptr;
787 m_sessionSaveWatcher = nullptr;
788 }
789 }
790
791 void DolphinMainWindow::saveProperties(KConfigGroup &group)
792 {
793 m_tabWidget->saveProperties(group);
794 }
795
796 void DolphinMainWindow::readProperties(const KConfigGroup &group)
797 {
798 m_tabWidget->readProperties(group);
799 }
800
801 void DolphinMainWindow::updateNewMenu()
802 {
803 m_newFileMenu->checkUpToDate();
804 m_newFileMenu->setWorkingDirectory(activeViewContainer()->url());
805 }
806
807 void DolphinMainWindow::createDirectory()
808 {
809 // When creating directory, namejob is being run. In network folders,
810 // this job can take long time, so instead of starting multiple namejobs,
811 // just check if we are already running one. This prevents opening multiple
812 // dialogs. BUG:481401
813 if (!m_newFileMenu->isCreateDirectoryRunning()) {
814 m_newFileMenu->setWorkingDirectory(activeViewContainer()->url());
815 m_newFileMenu->createDirectory();
816 }
817 }
818
819 void DolphinMainWindow::quit()
820 {
821 close();
822 }
823
824 void DolphinMainWindow::showErrorMessage(const QString &message)
825 {
826 m_activeViewContainer->showMessage(message, KMessageWidget::Error);
827 }
828
829 void DolphinMainWindow::slotUndoAvailable(bool available)
830 {
831 QAction *undoAction = actionCollection()->action(KStandardAction::name(KStandardAction::Undo));
832 if (undoAction) {
833 undoAction->setEnabled(available);
834 }
835 }
836
837 void DolphinMainWindow::slotUndoTextChanged(const QString &text)
838 {
839 QAction *undoAction = actionCollection()->action(KStandardAction::name(KStandardAction::Undo));
840 if (undoAction) {
841 undoAction->setText(text);
842 }
843 }
844
845 void DolphinMainWindow::undo()
846 {
847 clearStatusBar();
848 KIO::FileUndoManager::self()->uiInterface()->setParentWidget(this);
849 KIO::FileUndoManager::self()->undo();
850 }
851
852 void DolphinMainWindow::cut()
853 {
854 if (m_activeViewContainer->view()->selectedItems().isEmpty()) {
855 m_activeViewContainer->setSelectionModeEnabled(true, actionCollection(), SelectionMode::BottomBar::Contents::CutContents);
856 } else {
857 m_activeViewContainer->view()->cutSelectedItemsToClipboard();
858 m_activeViewContainer->setSelectionModeEnabled(false);
859 }
860 }
861
862 void DolphinMainWindow::copy()
863 {
864 if (m_activeViewContainer->view()->selectedItems().isEmpty()) {
865 m_activeViewContainer->setSelectionModeEnabled(true, actionCollection(), SelectionMode::BottomBar::Contents::CopyContents);
866 } else {
867 m_activeViewContainer->view()->copySelectedItemsToClipboard();
868 m_activeViewContainer->setSelectionModeEnabled(false);
869 }
870 }
871
872 void DolphinMainWindow::paste()
873 {
874 m_activeViewContainer->view()->paste();
875 }
876
877 void DolphinMainWindow::find()
878 {
879 m_activeViewContainer->setSearchModeEnabled(true);
880 }
881
882 void DolphinMainWindow::updateSearchAction()
883 {
884 QAction *toggleSearchAction = actionCollection()->action(QStringLiteral("toggle_search"));
885 toggleSearchAction->setChecked(m_activeViewContainer->isSearchModeEnabled());
886 }
887
888 void DolphinMainWindow::updatePasteAction()
889 {
890 QAction *pasteAction = actionCollection()->action(KStandardAction::name(KStandardAction::Paste));
891 QPair<bool, QString> pasteInfo = m_activeViewContainer->view()->pasteInfo();
892 pasteAction->setEnabled(pasteInfo.first);
893 m_disabledActionNotifier->setDisabledReason(pasteAction,
894 m_activeViewContainer->rootItem().isWritable()
895 ? i18nc("@info", "Cannot paste: The clipboard is empty.")
896 : i18nc("@info", "Cannot paste: You do not have permission to write into this folder."));
897 pasteAction->setText(pasteInfo.second);
898 }
899
900 void DolphinMainWindow::slotDirectoryLoadingCompleted()
901 {
902 updatePasteAction();
903 }
904
905 void DolphinMainWindow::slotToolBarActionMiddleClicked(QAction *action)
906 {
907 if (action == actionCollection()->action(KStandardAction::name(KStandardAction::Back))) {
908 goBackInNewTab();
909 } else if (action == actionCollection()->action(KStandardAction::name(KStandardAction::Forward))) {
910 goForwardInNewTab();
911 } else if (action == actionCollection()->action(QStringLiteral("go_up"))) {
912 goUpInNewTab();
913 } else if (action == actionCollection()->action(QStringLiteral("go_home"))) {
914 goHomeInNewTab();
915 }
916 }
917
918 QAction *DolphinMainWindow::urlNavigatorHistoryAction(const KUrlNavigator *urlNavigator, int historyIndex, QObject *parent)
919 {
920 const QUrl url = urlNavigator->locationUrl(historyIndex);
921
922 QString text = url.toDisplayString(QUrl::PreferLocalFile);
923
924 if (!urlNavigator->showFullPath()) {
925 const KFilePlacesModel *placesModel = DolphinPlacesModelSingleton::instance().placesModel();
926
927 const QModelIndex closestIdx = placesModel->closestItem(url);
928 if (closestIdx.isValid()) {
929 const QUrl placeUrl = placesModel->url(closestIdx);
930
931 text = placesModel->text(closestIdx);
932
933 QString pathInsidePlace = url.path().mid(placeUrl.path().length());
934
935 if (!pathInsidePlace.isEmpty() && !pathInsidePlace.startsWith(QLatin1Char('/'))) {
936 pathInsidePlace.prepend(QLatin1Char('/'));
937 }
938
939 if (pathInsidePlace != QLatin1Char('/')) {
940 text.append(pathInsidePlace);
941 }
942 }
943 }
944
945 QAction *action = new QAction(QIcon::fromTheme(KIO::iconNameForUrl(url)), text, parent);
946 action->setData(historyIndex);
947 return action;
948 }
949
950 void DolphinMainWindow::slotAboutToShowBackPopupMenu()
951 {
952 const KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
953 int entries = 0;
954 QMenu *menu = m_backAction->popupMenu();
955 menu->clear();
956 for (int i = urlNavigator->historyIndex() + 1; i < urlNavigator->historySize() && entries < MaxNumberOfNavigationentries; ++i, ++entries) {
957 QAction *action = urlNavigatorHistoryAction(urlNavigator, i, menu);
958 menu->addAction(action);
959 }
960 }
961
962 void DolphinMainWindow::slotGoBack(QAction *action)
963 {
964 int gotoIndex = action->data().value<int>();
965 const KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
966 for (int i = gotoIndex - urlNavigator->historyIndex(); i > 0; --i) {
967 goBack();
968 }
969 }
970
971 void DolphinMainWindow::slotBackForwardActionMiddleClicked(QAction *action)
972 {
973 if (action) {
974 const KUrlNavigator *urlNavigator = activeViewContainer()->urlNavigatorInternalWithHistory();
975 openNewTab(urlNavigator->locationUrl(action->data().value<int>()));
976 }
977 }
978
979 void DolphinMainWindow::slotAboutToShowForwardPopupMenu()
980 {
981 const KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
982 int entries = 0;
983 QMenu *menu = m_forwardAction->popupMenu();
984 for (int i = urlNavigator->historyIndex() - 1; i >= 0 && entries < MaxNumberOfNavigationentries; --i, ++entries) {
985 QAction *action = urlNavigatorHistoryAction(urlNavigator, i, menu);
986 menu->addAction(action);
987 }
988 }
989
990 void DolphinMainWindow::slotGoForward(QAction *action)
991 {
992 int gotoIndex = action->data().value<int>();
993 const KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
994 for (int i = urlNavigator->historyIndex() - gotoIndex; i > 0; --i) {
995 goForward();
996 }
997 }
998
999 void DolphinMainWindow::slotSetSelectionMode(bool enabled, SelectionMode::BottomBar::Contents bottomBarContents)
1000 {
1001 m_activeViewContainer->setSelectionModeEnabled(enabled, actionCollection(), bottomBarContents);
1002 }
1003
1004 void DolphinMainWindow::selectAll()
1005 {
1006 clearStatusBar();
1007
1008 // if the URL navigator is editable and focused, select the whole
1009 // URL instead of all items of the view
1010
1011 KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigator();
1012 QLineEdit *lineEdit = urlNavigator->editor()->lineEdit();
1013 const bool selectUrl = urlNavigator->isUrlEditable() && lineEdit->hasFocus();
1014 if (selectUrl) {
1015 lineEdit->selectAll();
1016 } else {
1017 m_activeViewContainer->view()->selectAll();
1018 }
1019 }
1020
1021 void DolphinMainWindow::invertSelection()
1022 {
1023 clearStatusBar();
1024 m_activeViewContainer->view()->invertSelection();
1025 }
1026
1027 void DolphinMainWindow::toggleSplitView()
1028 {
1029 DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
1030 tabPage->setSplitViewEnabled(!tabPage->splitViewEnabled(), WithAnimation);
1031 m_tabWidget->updateTabName(m_tabWidget->indexOf(tabPage));
1032 updateViewActions();
1033 }
1034
1035 void DolphinMainWindow::popoutSplitView()
1036 {
1037 DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
1038 if (!tabPage->splitViewEnabled())
1039 return;
1040 openNewWindow((GeneralSettings::closeActiveSplitView() ? tabPage->activeViewContainer() : tabPage->inactiveViewContainer())->url());
1041 tabPage->setSplitViewEnabled(false, WithAnimation);
1042 updateSplitActions();
1043 }
1044
1045 void DolphinMainWindow::toggleSplitStash()
1046 {
1047 DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
1048 tabPage->setSplitViewEnabled(false, WithAnimation);
1049 tabPage->setSplitViewEnabled(true, WithAnimation, QUrl("stash:/"));
1050 }
1051
1052 void DolphinMainWindow::copyToInactiveSplitView()
1053 {
1054 if (m_activeViewContainer->view()->selectedItems().isEmpty()) {
1055 m_activeViewContainer->setSelectionModeEnabled(true, actionCollection(), SelectionMode::BottomBar::Contents::CopyToOtherViewContents);
1056 } else {
1057 m_tabWidget->copyToInactiveSplitView();
1058 m_activeViewContainer->setSelectionModeEnabled(false);
1059 }
1060 }
1061
1062 void DolphinMainWindow::moveToInactiveSplitView()
1063 {
1064 if (m_activeViewContainer->view()->selectedItems().isEmpty()) {
1065 m_activeViewContainer->setSelectionModeEnabled(true, actionCollection(), SelectionMode::BottomBar::Contents::MoveToOtherViewContents);
1066 } else {
1067 m_tabWidget->moveToInactiveSplitView();
1068 m_activeViewContainer->setSelectionModeEnabled(false);
1069 }
1070 }
1071
1072 void DolphinMainWindow::reloadView()
1073 {
1074 clearStatusBar();
1075 m_activeViewContainer->reload();
1076 m_activeViewContainer->statusBar()->updateSpaceInfo();
1077 }
1078
1079 void DolphinMainWindow::stopLoading()
1080 {
1081 m_activeViewContainer->view()->stopLoading();
1082 }
1083
1084 void DolphinMainWindow::enableStopAction()
1085 {
1086 actionCollection()->action(QStringLiteral("stop"))->setEnabled(true);
1087 }
1088
1089 void DolphinMainWindow::disableStopAction()
1090 {
1091 actionCollection()->action(QStringLiteral("stop"))->setEnabled(false);
1092 }
1093
1094 void DolphinMainWindow::toggleSelectionMode()
1095 {
1096 const bool checked = !m_activeViewContainer->isSelectionModeEnabled();
1097
1098 m_activeViewContainer->setSelectionModeEnabled(checked, actionCollection(), SelectionMode::BottomBar::Contents::GeneralContents);
1099 actionCollection()->action(QStringLiteral("toggle_selection_mode"))->setChecked(checked);
1100 }
1101
1102 void DolphinMainWindow::showFilterBar()
1103 {
1104 m_activeViewContainer->setFilterBarVisible(true);
1105 }
1106
1107 void DolphinMainWindow::toggleFilterBar()
1108 {
1109 const bool checked = !m_activeViewContainer->isFilterBarVisible();
1110 m_activeViewContainer->setFilterBarVisible(checked);
1111
1112 QAction *toggleFilterBarAction = actionCollection()->action(QStringLiteral("toggle_filter"));
1113 toggleFilterBarAction->setChecked(checked);
1114 }
1115
1116 void DolphinMainWindow::toggleEditLocation()
1117 {
1118 clearStatusBar();
1119
1120 QAction *action = actionCollection()->action(QStringLiteral("editable_location"));
1121 KUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigator();
1122 urlNavigator->setUrlEditable(action->isChecked());
1123 }
1124
1125 void DolphinMainWindow::replaceLocation()
1126 {
1127 KUrlNavigator *navigator = m_activeViewContainer->urlNavigator();
1128 QLineEdit *lineEdit = navigator->editor()->lineEdit();
1129
1130 // If the text field currently has focus and everything is selected,
1131 // pressing the keyboard shortcut returns the whole thing to breadcrumb mode
1132 // and goes back to the view, just like how it was before this action was triggered the first time.
1133 if (navigator->isUrlEditable() && lineEdit->hasFocus() && lineEdit->selectedText() == lineEdit->text()) {
1134 navigator->setUrlEditable(false);
1135 m_activeViewContainer->view()->setFocus();
1136 } else {
1137 navigator->setUrlEditable(true);
1138 navigator->setFocus();
1139 lineEdit->selectAll();
1140 }
1141 }
1142
1143 void DolphinMainWindow::togglePanelLockState()
1144 {
1145 const bool newLockState = !GeneralSettings::lockPanels();
1146 const auto childrenObjects = children();
1147 for (QObject *child : childrenObjects) {
1148 DolphinDockWidget *dock = qobject_cast<DolphinDockWidget *>(child);
1149 if (dock) {
1150 dock->setLocked(newLockState);
1151 }
1152 }
1153
1154 DolphinPlacesModelSingleton::instance().placesModel()->setPanelsLocked(newLockState);
1155
1156 GeneralSettings::setLockPanels(newLockState);
1157 }
1158
1159 void DolphinMainWindow::slotTerminalPanelVisibilityChanged(bool visible)
1160 {
1161 if (!visible && m_activeViewContainer) {
1162 m_activeViewContainer->view()->setFocus();
1163 }
1164 // Putting focus to the Terminal is not handled here but in TerminalPanel::showEvent().
1165 }
1166
1167 void DolphinMainWindow::slotPlacesPanelVisibilityChanged(bool visible)
1168 {
1169 if (!visible && m_activeViewContainer) {
1170 m_activeViewContainer->view()->setFocus();
1171 return;
1172 }
1173 m_placesPanel->setFocus();
1174 }
1175
1176 void DolphinMainWindow::goBack()
1177 {
1178 DolphinUrlNavigator *urlNavigator = m_activeViewContainer->urlNavigatorInternalWithHistory();
1179 urlNavigator->goBack();
1180
1181 if (urlNavigator->locationState().isEmpty()) {
1182 // An empty location state indicates a redirection URL,
1183 // which must be skipped too
1184 urlNavigator->goBack();
1185 }
1186 }
1187
1188 void DolphinMainWindow::goForward()
1189 {
1190 m_activeViewContainer->urlNavigatorInternalWithHistory()->goForward();
1191 }
1192
1193 void DolphinMainWindow::goUp()
1194 {
1195 m_activeViewContainer->urlNavigatorInternalWithHistory()->goUp();
1196 }
1197
1198 void DolphinMainWindow::goHome()
1199 {
1200 m_activeViewContainer->urlNavigatorInternalWithHistory()->goHome();
1201 }
1202
1203 void DolphinMainWindow::goBackInNewTab()
1204 {
1205 const KUrlNavigator *urlNavigator = activeViewContainer()->urlNavigatorInternalWithHistory();
1206 const int index = urlNavigator->historyIndex() + 1;
1207 openNewTab(urlNavigator->locationUrl(index));
1208 }
1209
1210 void DolphinMainWindow::goForwardInNewTab()
1211 {
1212 const KUrlNavigator *urlNavigator = activeViewContainer()->urlNavigatorInternalWithHistory();
1213 const int index = urlNavigator->historyIndex() - 1;
1214 openNewTab(urlNavigator->locationUrl(index));
1215 }
1216
1217 void DolphinMainWindow::goUpInNewTab()
1218 {
1219 const QUrl currentUrl = activeViewContainer()->urlNavigator()->locationUrl();
1220 openNewTab(KIO::upUrl(currentUrl));
1221 }
1222
1223 void DolphinMainWindow::goHomeInNewTab()
1224 {
1225 openNewTab(Dolphin::homeUrl());
1226 }
1227
1228 void DolphinMainWindow::compareFiles()
1229 {
1230 const KFileItemList items = m_tabWidget->currentTabPage()->selectedItems();
1231 if (items.count() != 2) {
1232 // The action is disabled in this case, but it could have been triggered
1233 // via D-Bus, see https://bugs.kde.org/show_bug.cgi?id=325517
1234 return;
1235 }
1236
1237 QUrl urlA = items.at(0).url();
1238 QUrl urlB = items.at(1).url();
1239
1240 QString command(QStringLiteral("kompare -c \""));
1241 command.append(urlA.toDisplayString(QUrl::PreferLocalFile));
1242 command.append("\" \"");
1243 command.append(urlB.toDisplayString(QUrl::PreferLocalFile));
1244 command.append('\"');
1245
1246 KIO::CommandLauncherJob *job = new KIO::CommandLauncherJob(command, this);
1247 job->setDesktopName(QStringLiteral("org.kde.kompare"));
1248 job->start();
1249 }
1250
1251 void DolphinMainWindow::toggleShowMenuBar()
1252 {
1253 const bool visible = menuBar()->isVisible();
1254 menuBar()->setVisible(!visible);
1255 }
1256
1257 QPointer<QAction> DolphinMainWindow::preferredSearchTool()
1258 {
1259 m_searchTools.clear();
1260
1261 KService::Ptr kfind = KService::serviceByDesktopName(QStringLiteral("org.kde.kfind"));
1262
1263 if (!kfind) {
1264 return nullptr;
1265 }
1266
1267 auto *action = new QAction(QIcon::fromTheme(kfind->icon()), kfind->name(), this);
1268
1269 connect(action, &QAction::triggered, this, [this, kfind] {
1270 auto *job = new KIO::ApplicationLauncherJob(kfind);
1271 job->setUrls({m_activeViewContainer->url()});
1272 job->start();
1273 });
1274
1275 return action;
1276 }
1277
1278 void DolphinMainWindow::updateOpenPreferredSearchToolAction()
1279 {
1280 QAction *openPreferredSearchTool = actionCollection()->action(QStringLiteral("open_preferred_search_tool"));
1281 if (!openPreferredSearchTool) {
1282 return;
1283 }
1284 QPointer<QAction> tool = preferredSearchTool();
1285 if (tool) {
1286 openPreferredSearchTool->setVisible(true);
1287 openPreferredSearchTool->setText(i18nc("@action:inmenu Tools", "Open %1", tool->text()));
1288 // Only override with the app icon if it is the default, i.e. the user hasn't configured one manually
1289 // https://bugs.kde.org/show_bug.cgi?id=442815
1290 if (openPreferredSearchTool->icon().name() == QLatin1String("search")) {
1291 openPreferredSearchTool->setIcon(tool->icon());
1292 }
1293 } else {
1294 openPreferredSearchTool->setVisible(false);
1295 // still visible in Shortcuts configuration window
1296 openPreferredSearchTool->setText(i18nc("@action:inmenu Tools", "Open Preferred Search Tool"));
1297 openPreferredSearchTool->setIcon(QIcon::fromTheme(QStringLiteral("search")));
1298 }
1299 }
1300
1301 void DolphinMainWindow::openPreferredSearchTool()
1302 {
1303 QPointer<QAction> tool = preferredSearchTool();
1304 if (tool) {
1305 tool->trigger();
1306 }
1307 }
1308
1309 void DolphinMainWindow::openTerminal()
1310 {
1311 openTerminalJob(m_activeViewContainer->url());
1312 }
1313
1314 void DolphinMainWindow::openTerminalHere()
1315 {
1316 QList<QUrl> urls = {};
1317
1318 const auto selectedItems = m_activeViewContainer->view()->selectedItems();
1319 for (const KFileItem &item : selectedItems) {
1320 QUrl url = item.targetUrl();
1321 if (item.isFile()) {
1322 url.setPath(QFileInfo(url.path()).absolutePath());
1323 }
1324 if (!urls.contains(url)) {
1325 urls << url;
1326 }
1327 }
1328
1329 // No items are selected. Open a terminal window for the current location.
1330 if (urls.count() == 0) {
1331 openTerminal();
1332 return;
1333 }
1334
1335 if (urls.count() > 5) {
1336 QString question = i18np("Are you sure you want to open 1 terminal window?", "Are you sure you want to open %1 terminal windows?", urls.count());
1337 const int answer = KMessageBox::warningContinueCancel(
1338 this,
1339 question,
1340 {},
1341 KGuiItem(i18ncp("@action:button", "Open %1 Terminal", "Open %1 Terminals", urls.count()), QStringLiteral("utilities-terminal")),
1342 KStandardGuiItem::cancel(),
1343 QStringLiteral("ConfirmOpenManyTerminals"));
1344 if (answer != KMessageBox::PrimaryAction && answer != KMessageBox::Continue) {
1345 return;
1346 }
1347 }
1348
1349 for (const QUrl &url : std::as_const(urls)) {
1350 openTerminalJob(url);
1351 }
1352 }
1353
1354 void DolphinMainWindow::openTerminalJob(const QUrl &url)
1355 {
1356 if (url.isLocalFile()) {
1357 auto job = new KTerminalLauncherJob(QString());
1358 job->setWorkingDirectory(url.toLocalFile());
1359 job->start();
1360 return;
1361 }
1362
1363 // Not a local file, with protocol Class ":local", try stat'ing
1364 if (KProtocolInfo::protocolClass(url.scheme()) == QLatin1String(":local")) {
1365 KIO::StatJob *job = KIO::mostLocalUrl(url);
1366 KJobWidgets::setWindow(job, this);
1367 connect(job, &KJob::result, this, [job]() {
1368 QUrl statUrl;
1369 if (!job->error()) {
1370 statUrl = job->mostLocalUrl();
1371 }
1372
1373 auto job = new KTerminalLauncherJob(QString());
1374 job->setWorkingDirectory(statUrl.isLocalFile() ? statUrl.toLocalFile() : QDir::homePath());
1375 job->start();
1376 });
1377
1378 return;
1379 }
1380
1381 // Nothing worked, just use $HOME
1382 auto job = new KTerminalLauncherJob(QString());
1383 job->setWorkingDirectory(QDir::homePath());
1384 job->start();
1385 }
1386
1387 void DolphinMainWindow::editSettings()
1388 {
1389 if (!m_settingsDialog) {
1390 DolphinViewContainer *container = activeViewContainer();
1391 container->view()->writeSettings();
1392
1393 const QUrl url = container->url();
1394 DolphinSettingsDialog *settingsDialog = new DolphinSettingsDialog(url, this, actionCollection());
1395 connect(settingsDialog, &DolphinSettingsDialog::settingsChanged, this, &DolphinMainWindow::refreshViews);
1396 connect(settingsDialog, &DolphinSettingsDialog::settingsChanged, &DolphinUrlNavigatorsController::slotReadSettings);
1397 settingsDialog->setAttribute(Qt::WA_DeleteOnClose);
1398 settingsDialog->show();
1399 m_settingsDialog = settingsDialog;
1400 } else {
1401 m_settingsDialog.data()->raise();
1402 }
1403 }
1404
1405 void DolphinMainWindow::handleUrl(const QUrl &url)
1406 {
1407 delete m_lastHandleUrlOpenJob;
1408 m_lastHandleUrlOpenJob = nullptr;
1409
1410 if (url.isLocalFile() && QFileInfo(url.toLocalFile()).isDir()) {
1411 activeViewContainer()->setUrl(url);
1412 } else {
1413 m_lastHandleUrlOpenJob = new KIO::OpenUrlJob(url);
1414 m_lastHandleUrlOpenJob->setUiDelegate(KIO::createDefaultJobUiDelegate(KJobUiDelegate::AutoHandlingEnabled, this));
1415 m_lastHandleUrlOpenJob->setShowOpenOrExecuteDialog(true);
1416
1417 connect(m_lastHandleUrlOpenJob, &KIO::OpenUrlJob::mimeTypeFound, this, [this, url](const QString &mimetype) {
1418 if (mimetype == QLatin1String("inode/directory")) {
1419 // If it's a dir, we'll take it from here
1420 m_lastHandleUrlOpenJob->kill();
1421 m_lastHandleUrlOpenJob = nullptr;
1422 activeViewContainer()->setUrl(url);
1423 }
1424 });
1425
1426 connect(m_lastHandleUrlOpenJob, &KIO::OpenUrlJob::result, this, [this]() {
1427 m_lastHandleUrlOpenJob = nullptr;
1428 });
1429
1430 m_lastHandleUrlOpenJob->start();
1431 }
1432 }
1433
1434 void DolphinMainWindow::slotWriteStateChanged(bool isFolderWritable)
1435 {
1436 // trash:/ is writable but we don't want to create new items in it.
1437 // TODO: remove the trash check once https://phabricator.kde.org/T8234 is implemented
1438 newFileMenu()->setEnabled(isFolderWritable && m_activeViewContainer->url().scheme() != QLatin1String("trash"));
1439 // When the menu is disabled, actions in it are disabled later in the event loop, and we need to set the disabled reason after that.
1440 QTimer::singleShot(0, this, [this]() {
1441 m_disabledActionNotifier->setDisabledReason(actionCollection()->action(QStringLiteral("create_dir")),
1442 i18nc("@info", "Cannot create new folder: You do not have permission to create items in this folder."));
1443 });
1444 }
1445
1446 void DolphinMainWindow::openContextMenu(const QPoint &pos, const KFileItem &item, const KFileItemList &selectedItems, const QUrl &url)
1447 {
1448 QPointer<DolphinContextMenu> contextMenu = new DolphinContextMenu(this, item, selectedItems, url, &m_fileItemActions);
1449 contextMenu->exec(pos);
1450
1451 // Delete the menu, unless it has been deleted in its own nested event loop already.
1452 if (contextMenu) {
1453 contextMenu->deleteLater();
1454 }
1455 }
1456
1457 QMenu *DolphinMainWindow::createPopupMenu()
1458 {
1459 QMenu *menu = KXmlGuiWindow::createPopupMenu();
1460
1461 menu->addSeparator();
1462 menu->addAction(actionCollection()->action(QStringLiteral("lock_panels")));
1463
1464 return menu;
1465 }
1466
1467 void DolphinMainWindow::updateHamburgerMenu()
1468 {
1469 KActionCollection *ac = actionCollection();
1470 auto hamburgerMenu = static_cast<KHamburgerMenu *>(ac->action(KStandardAction::name(KStandardAction::HamburgerMenu)));
1471 auto menu = hamburgerMenu->menu();
1472 if (!menu) {
1473 menu = new QMenu(this);
1474 hamburgerMenu->setMenu(menu);
1475 hamburgerMenu->hideActionsOf(ac->action(QStringLiteral("basic_actions"))->menu());
1476 hamburgerMenu->hideActionsOf(ac->action(QStringLiteral("zoom"))->menu());
1477 } else {
1478 menu->clear();
1479 }
1480 const QList<QAction *> toolbarActions = toolBar()->actions();
1481
1482 if (!toolBar()->isVisible()) {
1483 // If neither the menu bar nor the toolbar are visible, these actions should be available.
1484 menu->addAction(ac->action(KStandardAction::name(KStandardAction::ShowMenubar)));
1485 menu->addAction(toolBarMenuAction());
1486 menu->addSeparator();
1487 }
1488
1489 // This group of actions (until the next separator) contains all the most basic actions
1490 // necessary to use Dolphin effectively.
1491 menu->addAction(ac->action(QStringLiteral("go_back")));
1492 menu->addAction(ac->action(QStringLiteral("go_forward")));
1493
1494 menu->addMenu(m_newFileMenu->menu());
1495 if (!toolBar()->isVisible() || !toolbarActions.contains(ac->action(QStringLiteral("toggle_selection_mode_tool_bar")))) {
1496 menu->addAction(ac->action(QStringLiteral("toggle_selection_mode")));
1497 }
1498 menu->addAction(ac->action(QStringLiteral("basic_actions")));
1499 menu->addAction(ac->action(KStandardAction::name(KStandardAction::Undo)));
1500 if (!toolBar()->isVisible()
1501 || (!toolbarActions.contains(ac->action(QStringLiteral("toggle_search")))
1502 && !toolbarActions.contains(ac->action(QStringLiteral("open_preferred_search_tool"))))) {
1503 menu->addAction(ac->action(KStandardAction::name(KStandardAction::Find)));
1504 // This way a search action will only be added if none of the three available
1505 // search actions is present on the toolbar.
1506 }
1507 if (!toolBar()->isVisible() || !toolbarActions.contains(ac->action(QStringLiteral("toggle_filter")))) {
1508 menu->addAction(ac->action(QStringLiteral("show_filter_bar")));
1509 // This way a filter action will only be added if none of the two available
1510 // filter actions is present on the toolbar.
1511 }
1512 menu->addSeparator();
1513
1514 // The second group of actions (up until the next separator) contains actions for opening
1515 // additional views to interact with the file system.
1516 menu->addAction(ac->action(QStringLiteral("file_new")));
1517 menu->addAction(ac->action(QStringLiteral("new_tab")));
1518 if (ac->action(QStringLiteral("undo_close_tab"))->isEnabled()) {
1519 menu->addAction(ac->action(QStringLiteral("closed_tabs")));
1520 }
1521 menu->addAction(ac->action(QStringLiteral("open_terminal")));
1522 menu->addSeparator();
1523
1524 // The third group contains actions to change what one sees in the view
1525 // and to change the more general UI.
1526 if (!toolBar()->isVisible()
1527 || (!toolbarActions.contains(ac->action(QStringLiteral("icons"))) && !toolbarActions.contains(ac->action(QStringLiteral("compact")))
1528 && !toolbarActions.contains(ac->action(QStringLiteral("details"))) && !toolbarActions.contains(ac->action(QStringLiteral("view_mode"))))) {
1529 menu->addAction(ac->action(QStringLiteral("view_mode")));
1530 }
1531 menu->addAction(ac->action(QStringLiteral("show_hidden_files")));
1532 menu->addAction(ac->action(QStringLiteral("sort")));
1533 menu->addAction(ac->action(QStringLiteral("additional_info")));
1534 if (!GeneralSettings::showStatusBar() || !GeneralSettings::showZoomSlider()) {
1535 menu->addAction(ac->action(QStringLiteral("zoom")));
1536 }
1537 menu->addAction(ac->action(QStringLiteral("panels")));
1538
1539 // The "Configure" menu is not added to the actionCollection() because there is hardly
1540 // a good reason for users to put it on their toolbar.
1541 auto configureMenu = menu->addMenu(QIcon::fromTheme(QStringLiteral("configure")), i18nc("@action:inmenu menu for configure actions", "Configure"));
1542 configureMenu->addAction(ac->action(KStandardAction::name(KStandardAction::SwitchApplicationLanguage)));
1543 configureMenu->addAction(ac->action(KStandardAction::name(KStandardAction::KeyBindings)));
1544 configureMenu->addAction(ac->action(KStandardAction::name(KStandardAction::ConfigureToolbars)));
1545 configureMenu->addAction(ac->action(KStandardAction::name(KStandardAction::Preferences)));
1546 hamburgerMenu->hideActionsOf(configureMenu);
1547 }
1548
1549 void DolphinMainWindow::slotPlaceActivated(const QUrl &url)
1550 {
1551 DolphinViewContainer *view = activeViewContainer();
1552
1553 if (view->url() == url) {
1554 view->clearFilterBar(); // Fixes bug 259382.
1555
1556 // We can end up here if the user clicked a device in the Places Panel
1557 // which had been unmounted earlier, see https://bugs.kde.org/show_bug.cgi?id=161385.
1558 reloadView();
1559 } else {
1560 view->disableUrlNavigatorSelectionRequests();
1561 changeUrl(url);
1562 view->enableUrlNavigatorSelectionRequests();
1563 }
1564 }
1565
1566 void DolphinMainWindow::closedTabsCountChanged(unsigned int count)
1567 {
1568 actionCollection()->action(QStringLiteral("undo_close_tab"))->setEnabled(count > 0);
1569 }
1570
1571 void DolphinMainWindow::activeViewChanged(DolphinViewContainer *viewContainer)
1572 {
1573 DolphinViewContainer *oldViewContainer = m_activeViewContainer;
1574 Q_ASSERT(viewContainer);
1575
1576 m_activeViewContainer = viewContainer;
1577
1578 if (oldViewContainer) {
1579 const QAction *toggleSearchAction = actionCollection()->action(QStringLiteral("toggle_search"));
1580 toggleSearchAction->disconnect(oldViewContainer);
1581
1582 // Disconnect all signals between the old view container (container,
1583 // view and url navigator) and main window.
1584 oldViewContainer->disconnect(this);
1585 oldViewContainer->view()->disconnect(this);
1586 oldViewContainer->urlNavigatorInternalWithHistory()->disconnect(this);
1587 auto navigators = static_cast<DolphinNavigatorsWidgetAction *>(actionCollection()->action(QStringLiteral("url_navigators")));
1588 navigators->primaryUrlNavigator()->disconnect(this);
1589 if (auto secondaryUrlNavigator = navigators->secondaryUrlNavigator()) {
1590 secondaryUrlNavigator->disconnect(this);
1591 }
1592
1593 // except the requestItemInfo so that on hover the information panel can still be updated
1594 connect(oldViewContainer->view(), &DolphinView::requestItemInfo, this, &DolphinMainWindow::requestItemInfo);
1595
1596 // Disconnect other slots.
1597 disconnect(oldViewContainer,
1598 &DolphinViewContainer::selectionModeChanged,
1599 actionCollection()->action(QStringLiteral("toggle_selection_mode")),
1600 &QAction::setChecked);
1601 }
1602
1603 connectViewSignals(viewContainer);
1604
1605 m_actionHandler->setCurrentView(viewContainer->view());
1606
1607 updateHistory();
1608 updateFileAndEditActions();
1609 updatePasteAction();
1610 updateViewActions();
1611 updateGoActions();
1612 updateSearchAction();
1613
1614 const QUrl url = viewContainer->url();
1615 Q_EMIT urlChanged(url);
1616 }
1617
1618 void DolphinMainWindow::tabCountChanged(int count)
1619 {
1620 const bool enableTabActions = (count > 1);
1621 for (int i = 0; i < MaxActivateTabShortcuts; ++i) {
1622 actionCollection()->action(QStringLiteral("activate_tab_%1").arg(i))->setEnabled(enableTabActions);
1623 }
1624 actionCollection()->action(QStringLiteral("activate_last_tab"))->setEnabled(enableTabActions);
1625 actionCollection()->action(QStringLiteral("activate_next_tab"))->setEnabled(enableTabActions);
1626 actionCollection()->action(QStringLiteral("activate_prev_tab"))->setEnabled(enableTabActions);
1627 }
1628
1629 void DolphinMainWindow::updateWindowTitle()
1630 {
1631 const QString newTitle = m_activeViewContainer->captionWindowTitle();
1632 if (windowTitle() != newTitle) {
1633 setWindowTitle(newTitle);
1634 }
1635 }
1636
1637 void DolphinMainWindow::slotStorageTearDownFromPlacesRequested(const QString &mountPath)
1638 {
1639 connect(m_placesPanel, &PlacesPanel::storageTearDownSuccessful, this, [this, mountPath]() {
1640 setViewsToHomeIfMountPathOpen(mountPath);
1641 });
1642
1643 if (m_terminalPanel && m_terminalPanel->currentWorkingDirectoryIsChildOf(mountPath)) {
1644 m_tearDownFromPlacesRequested = true;
1645 m_terminalPanel->goHome();
1646 // m_placesPanel->proceedWithTearDown() will be called in slotTerminalDirectoryChanged
1647 } else {
1648 m_placesPanel->proceedWithTearDown();
1649 }
1650 }
1651
1652 void DolphinMainWindow::slotStorageTearDownExternallyRequested(const QString &mountPath)
1653 {
1654 connect(m_placesPanel, &PlacesPanel::storageTearDownSuccessful, this, [this, mountPath]() {
1655 setViewsToHomeIfMountPathOpen(mountPath);
1656 });
1657
1658 if (m_terminalPanel && m_terminalPanel->currentWorkingDirectoryIsChildOf(mountPath)) {
1659 m_tearDownFromPlacesRequested = false;
1660 m_terminalPanel->goHome();
1661 }
1662 }
1663
1664 void DolphinMainWindow::slotKeyBindings()
1665 {
1666 KShortcutsDialog dialog(KShortcutsEditor::AllActions, KShortcutsEditor::LetterShortcutsAllowed, this);
1667 dialog.addCollection(actionCollection());
1668 if (m_terminalPanel) {
1669 KActionCollection *konsolePartActionCollection = m_terminalPanel->actionCollection();
1670 if (konsolePartActionCollection) {
1671 dialog.addCollection(konsolePartActionCollection, QStringLiteral("KonsolePart"));
1672 }
1673 }
1674 dialog.configure();
1675 }
1676
1677 void DolphinMainWindow::setViewsToHomeIfMountPathOpen(const QString &mountPath)
1678 {
1679 const QVector<DolphinViewContainer *> theViewContainers = viewContainers();
1680 for (DolphinViewContainer *viewContainer : theViewContainers) {
1681 if (viewContainer && viewContainer->url().toLocalFile().startsWith(mountPath)) {
1682 viewContainer->setUrl(QUrl::fromLocalFile(QDir::homePath()));
1683 }
1684 }
1685 disconnect(m_placesPanel, &PlacesPanel::storageTearDownSuccessful, nullptr, nullptr);
1686 }
1687
1688 void DolphinMainWindow::setupActions()
1689 {
1690 auto hamburgerMenuAction = KStandardAction::hamburgerMenu(nullptr, nullptr, actionCollection());
1691
1692 // setup 'File' menu
1693 m_newFileMenu = new DolphinNewFileMenu(nullptr, this);
1694 actionCollection()->addAction(QStringLiteral("new_menu"), m_newFileMenu);
1695 QMenu *menu = m_newFileMenu->menu();
1696 menu->setTitle(i18nc("@title:menu Create new folder, file, link, etc.", "Create New"));
1697 menu->setIcon(QIcon::fromTheme(QStringLiteral("list-add")));
1698 m_newFileMenu->setPopupMode(QToolButton::InstantPopup);
1699 connect(menu, &QMenu::aboutToShow, this, &DolphinMainWindow::updateNewMenu);
1700
1701 QAction *newWindow = KStandardAction::openNew(this, &DolphinMainWindow::openNewMainWindow, actionCollection());
1702 newWindow->setText(i18nc("@action:inmenu File", "New &Window"));
1703 newWindow->setToolTip(i18nc("@info", "Open a new Dolphin window"));
1704 newWindow->setWhatsThis(xi18nc("@info:whatsthis",
1705 "This opens a new "
1706 "window just like this one with the current location."
1707 "<nl/>You can drag and drop items between windows."));
1708 newWindow->setIcon(QIcon::fromTheme(QStringLiteral("window-new")));
1709
1710 QAction *newTab = actionCollection()->addAction(QStringLiteral("new_tab"));
1711 newTab->setIcon(QIcon::fromTheme(QStringLiteral("tab-new")));
1712 newTab->setText(i18nc("@action:inmenu File", "New Tab"));
1713 newTab->setWhatsThis(xi18nc("@info:whatsthis",
1714 "This opens a new "
1715 "<emphasis>Tab</emphasis> with the current location."
1716 "<nl/>Tabs allow you to quickly switch between multiple locations and views within this window. "
1717 "You can drag and drop items between tabs."));
1718 actionCollection()->setDefaultShortcut(newTab, Qt::CTRL | Qt::Key_T);
1719 connect(newTab, &QAction::triggered, this, &DolphinMainWindow::openNewActivatedTab);
1720
1721 QAction *addToPlaces = actionCollection()->addAction(QStringLiteral("add_to_places"));
1722 addToPlaces->setIcon(QIcon::fromTheme(QStringLiteral("bookmark-new")));
1723 addToPlaces->setText(i18nc("@action:inmenu Add current folder to places", "Add to Places"));
1724 addToPlaces->setWhatsThis(xi18nc("@info:whatsthis",
1725 "This adds the selected folder "
1726 "to the Places panel."));
1727 connect(addToPlaces, &QAction::triggered, this, &DolphinMainWindow::addToPlaces);
1728
1729 QAction *closeTab = KStandardAction::close(m_tabWidget, QOverload<>::of(&DolphinTabWidget::closeTab), actionCollection());
1730 closeTab->setText(i18nc("@action:inmenu File", "Close Tab"));
1731 closeTab->setToolTip(i18nc("@info", "Close Tab"));
1732 closeTab->setWhatsThis(i18nc("@info:whatsthis",
1733 "This closes the "
1734 "currently viewed tab. If no more tabs are left, this closes "
1735 "the whole window instead."));
1736
1737 QAction *quitAction = KStandardAction::quit(this, &DolphinMainWindow::quit, actionCollection());
1738 quitAction->setWhatsThis(i18nc("@info:whatsthis quit", "This closes this window."));
1739
1740 // setup 'Edit' menu
1741 KStandardAction::undo(this, &DolphinMainWindow::undo, actionCollection());
1742
1743 // i18n: This will be the last paragraph for the whatsthis for all three:
1744 // Cut, Copy and Paste
1745 const QString cutCopyPastePara = xi18nc("@info:whatsthis",
1746 "<para><emphasis>Cut, "
1747 "Copy</emphasis> and <emphasis>Paste</emphasis> work between many "
1748 "applications and are among the most used commands. That's why their "
1749 "<emphasis>keyboard shortcuts</emphasis> are prominently placed right "
1750 "next to each other on the keyboard: <shortcut>Ctrl+X</shortcut>, "
1751 "<shortcut>Ctrl+C</shortcut> and <shortcut>Ctrl+V</shortcut>.</para>");
1752 QAction *cutAction = KStandardAction::cut(this, &DolphinMainWindow::cut, actionCollection());
1753 m_actionTextHelper->registerTextWhenNothingIsSelected(cutAction, i18nc("@action", "Cut…"));
1754 cutAction->setWhatsThis(xi18nc("@info:whatsthis cut",
1755 "This copies the items "
1756 "in your current selection to the <emphasis>clipboard</emphasis>.<nl/>"
1757 "Use the <emphasis>Paste</emphasis> action afterwards to copy them from "
1758 "the clipboard to a new location. The items will be removed from their "
1759 "initial location.")
1760 + cutCopyPastePara);
1761 QAction *copyAction = KStandardAction::copy(this, &DolphinMainWindow::copy, actionCollection());
1762 m_actionTextHelper->registerTextWhenNothingIsSelected(copyAction, i18nc("@action", "Copy…"));
1763 copyAction->setWhatsThis(xi18nc("@info:whatsthis copy",
1764 "This copies the "
1765 "items in your current selection to the <emphasis>clipboard</emphasis>."
1766 "<nl/>Use the <emphasis>Paste</emphasis> action afterwards to copy them "
1767 "from the clipboard to a new location.")
1768 + cutCopyPastePara);
1769 QAction *paste = KStandardAction::paste(this, &DolphinMainWindow::paste, actionCollection());
1770 // The text of the paste-action is modified dynamically by Dolphin
1771 // (e. g. to "Paste One Folder"). To prevent that the size of the toolbar changes
1772 // due to the long text, the text "Paste" is used:
1773 paste->setIconText(i18nc("@action:inmenu Edit", "Paste"));
1774 paste->setWhatsThis(xi18nc("@info:whatsthis paste",
1775 "This copies the items from "
1776 "your <emphasis>clipboard</emphasis> to the currently viewed folder.<nl/>"
1777 "If the items were added to the clipboard by the <emphasis>Cut</emphasis> "
1778 "action they are removed from their old location.")
1779 + cutCopyPastePara);
1780
1781 QAction *copyToOtherViewAction = actionCollection()->addAction(QStringLiteral("copy_to_inactive_split_view"));
1782 copyToOtherViewAction->setText(i18nc("@action:inmenu", "Copy to Other View"));
1783 m_actionTextHelper->registerTextWhenNothingIsSelected(copyToOtherViewAction, i18nc("@action:inmenu", "Copy to Other View…"));
1784 copyToOtherViewAction->setWhatsThis(xi18nc("@info:whatsthis Copy",
1785 "This copies the selected items from "
1786 "the view in focus to the other view. "
1787 "(Only available while in Split View mode.)"));
1788 copyToOtherViewAction->setIcon(QIcon::fromTheme(QStringLiteral("edit-copy")));
1789 copyToOtherViewAction->setIconText(i18nc("@action:inmenu Edit", "Copy to Other View"));
1790 actionCollection()->setDefaultShortcut(copyToOtherViewAction, Qt::SHIFT | Qt::Key_F5);
1791 connect(copyToOtherViewAction, &QAction::triggered, this, &DolphinMainWindow::copyToInactiveSplitView);
1792
1793 QAction *moveToOtherViewAction = actionCollection()->addAction(QStringLiteral("move_to_inactive_split_view"));
1794 moveToOtherViewAction->setText(i18nc("@action:inmenu", "Move to Other View"));
1795 m_actionTextHelper->registerTextWhenNothingIsSelected(moveToOtherViewAction, i18nc("@action:inmenu", "Move to Other View…"));
1796 moveToOtherViewAction->setWhatsThis(xi18nc("@info:whatsthis Move",
1797 "This moves the selected items from "
1798 "the view in focus to the other view. "
1799 "(Only available while in Split View mode.)"));
1800 moveToOtherViewAction->setIcon(QIcon::fromTheme(QStringLiteral("edit-cut")));
1801 moveToOtherViewAction->setIconText(i18nc("@action:inmenu Edit", "Move to Other View"));
1802 actionCollection()->setDefaultShortcut(moveToOtherViewAction, Qt::SHIFT | Qt::Key_F6);
1803 connect(moveToOtherViewAction, &QAction::triggered, this, &DolphinMainWindow::moveToInactiveSplitView);
1804
1805 QAction *showFilterBar = actionCollection()->addAction(QStringLiteral("show_filter_bar"));
1806 showFilterBar->setText(i18nc("@action:inmenu Tools", "Filter…"));
1807 showFilterBar->setToolTip(i18nc("@info:tooltip", "Show Filter Bar"));
1808 showFilterBar->setWhatsThis(xi18nc("@info:whatsthis",
1809 "This opens the "
1810 "<emphasis>Filter Bar</emphasis> at the bottom of the window.<nl/> "
1811 "There you can enter text to filter the files and folders currently displayed. "
1812 "Only those that contain the text in their name will be kept in view."));
1813 showFilterBar->setIcon(QIcon::fromTheme(QStringLiteral("view-filter")));
1814 actionCollection()->setDefaultShortcuts(showFilterBar, {Qt::CTRL | Qt::Key_I, Qt::Key_Slash});
1815 connect(showFilterBar, &QAction::triggered, this, &DolphinMainWindow::showFilterBar);
1816
1817 // toggle_filter acts as a copy of the main showFilterBar to be used mainly
1818 // in the toolbar, with no default shortcut attached, to avoid messing with
1819 // existing workflows (filter bar always open and Ctrl-I to focus)
1820 QAction *toggleFilter = actionCollection()->addAction(QStringLiteral("toggle_filter"));
1821 toggleFilter->setText(i18nc("@action:inmenu", "Toggle Filter Bar"));
1822 toggleFilter->setIconText(i18nc("@action:intoolbar", "Filter"));
1823 toggleFilter->setIcon(showFilterBar->icon());
1824 toggleFilter->setToolTip(showFilterBar->toolTip());
1825 toggleFilter->setWhatsThis(showFilterBar->whatsThis());
1826 toggleFilter->setCheckable(true);
1827 connect(toggleFilter, &QAction::triggered, this, &DolphinMainWindow::toggleFilterBar);
1828
1829 QAction *searchAction = KStandardAction::find(this, &DolphinMainWindow::find, actionCollection());
1830 searchAction->setText(i18n("Search…"));
1831 searchAction->setToolTip(i18nc("@info:tooltip", "Search for files and folders"));
1832 searchAction->setWhatsThis(xi18nc("@info:whatsthis find",
1833 "<para>This helps you "
1834 "find files and folders by opening a <emphasis>search bar</emphasis>. "
1835 "There you can enter search terms and specify settings to find the "
1836 "items you are looking for.</para><para>Use this help again on "
1837 "the search bar so we can have a look at it while the settings are "
1838 "explained.</para>"));
1839
1840 // toggle_search acts as a copy of the main searchAction to be used mainly
1841 // in the toolbar, with no default shortcut attached, to avoid messing with
1842 // existing workflows (search bar always open and Ctrl-F to focus)
1843 QAction *toggleSearchAction = actionCollection()->addAction(QStringLiteral("toggle_search"));
1844 toggleSearchAction->setText(i18nc("@action:inmenu", "Toggle Search Bar"));
1845 toggleSearchAction->setIconText(i18nc("@action:intoolbar", "Search"));
1846 toggleSearchAction->setIcon(searchAction->icon());
1847 toggleSearchAction->setToolTip(searchAction->toolTip());
1848 toggleSearchAction->setWhatsThis(searchAction->whatsThis());
1849 toggleSearchAction->setCheckable(true);
1850
1851 QAction *toggleSelectionModeAction = actionCollection()->addAction(QStringLiteral("toggle_selection_mode"));
1852 // i18n: This action toggles a selection mode.
1853 toggleSelectionModeAction->setText(i18nc("@action:inmenu", "Select Files and Folders"));
1854 // i18n: Opens a selection mode for selecting files/folders.
1855 // The text is kept so unspecific because it will be shown on the toolbar where space is at a premium.
1856 toggleSelectionModeAction->setIconText(i18nc("@action:intoolbar", "Select"));
1857 toggleSelectionModeAction->setWhatsThis(xi18nc(
1858 "@info:whatsthis",
1859 "<para>This application only knows which files or folders should be acted on if they are"
1860 " <emphasis>selected</emphasis> first. Press this to toggle a <emphasis>Selection Mode</emphasis> which makes selecting and deselecting as easy as "
1861 "pressing an item once.</para><para>While in this mode, a quick access bar at the bottom shows available actions for the currently selected items."
1862 "</para>"));
1863 toggleSelectionModeAction->setIcon(QIcon::fromTheme(QStringLiteral("quickwizard")));
1864 toggleSelectionModeAction->setCheckable(true);
1865 actionCollection()->setDefaultShortcut(toggleSelectionModeAction, Qt::Key_Space);
1866 connect(toggleSelectionModeAction, &QAction::triggered, this, &DolphinMainWindow::toggleSelectionMode);
1867
1868 // A special version of the toggleSelectionModeAction for the toolbar that also contains a menu
1869 // with the selectAllAction and invertSelectionAction.
1870 auto *toggleSelectionModeToolBarAction =
1871 new KToolBarPopupAction(toggleSelectionModeAction->icon(), toggleSelectionModeAction->iconText(), actionCollection());
1872 toggleSelectionModeToolBarAction->setToolTip(toggleSelectionModeAction->text());
1873 toggleSelectionModeToolBarAction->setWhatsThis(toggleSelectionModeAction->whatsThis());
1874 actionCollection()->addAction(QStringLiteral("toggle_selection_mode_tool_bar"), toggleSelectionModeToolBarAction);
1875 toggleSelectionModeToolBarAction->setCheckable(true);
1876 toggleSelectionModeToolBarAction->setPopupMode(KToolBarPopupAction::DelayedPopup);
1877 connect(toggleSelectionModeToolBarAction, &QAction::triggered, toggleSelectionModeAction, &QAction::trigger);
1878 connect(toggleSelectionModeAction, &QAction::toggled, toggleSelectionModeToolBarAction, &QAction::setChecked);
1879
1880 QAction *selectAllAction = KStandardAction::selectAll(this, &DolphinMainWindow::selectAll, actionCollection());
1881 selectAllAction->setWhatsThis(xi18nc("@info:whatsthis",
1882 "This selects all "
1883 "files and folders in the current location."));
1884
1885 QAction *invertSelection = actionCollection()->addAction(QStringLiteral("invert_selection"));
1886 invertSelection->setText(i18nc("@action:inmenu Edit", "Invert Selection"));
1887 invertSelection->setWhatsThis(xi18nc("@info:whatsthis invert",
1888 "This selects all "
1889 "items that you have currently <emphasis>not</emphasis> selected instead."));
1890 invertSelection->setIcon(QIcon::fromTheme(QStringLiteral("edit-select-invert")));
1891 actionCollection()->setDefaultShortcut(invertSelection, Qt::CTRL | Qt::SHIFT | Qt::Key_A);
1892 connect(invertSelection, &QAction::triggered, this, &DolphinMainWindow::invertSelection);
1893
1894 QMenu *toggleSelectionModeActionMenu = new QMenu(this);
1895 toggleSelectionModeActionMenu->addAction(selectAllAction);
1896 toggleSelectionModeActionMenu->addAction(invertSelection);
1897 toggleSelectionModeToolBarAction->setMenu(toggleSelectionModeActionMenu);
1898
1899 // setup 'View' menu
1900 // (note that most of it is set up in DolphinViewActionHandler)
1901
1902 Admin::WorkerIntegration::createActAsAdminAction(actionCollection(), this);
1903
1904 m_splitViewAction = actionCollection()->add<KActionMenu>(QStringLiteral("split_view"));
1905 m_splitViewMenuAction = actionCollection()->addAction(QStringLiteral("split_view_menu"));
1906
1907 m_splitViewAction->setWhatsThis(xi18nc("@info:whatsthis split",
1908 "<para>This presents "
1909 "a second view side-by-side with the current view, so you can see "
1910 "the contents of two folders at once and easily move items between "
1911 "them.</para><para>The view that is not \"in focus\" will be dimmed. "
1912 "</para>Click this button again to close one of the views."));
1913 m_splitViewMenuAction->setWhatsThis(m_splitViewAction->whatsThis());
1914
1915 // only set it for the menu version
1916 actionCollection()->setDefaultShortcut(m_splitViewMenuAction, Qt::Key_F3);
1917
1918 connect(m_splitViewAction, &QAction::triggered, this, &DolphinMainWindow::toggleSplitView);
1919 connect(m_splitViewMenuAction, &QAction::triggered, this, &DolphinMainWindow::toggleSplitView);
1920
1921 QAction *popoutSplit = actionCollection()->addAction(QStringLiteral("popout_split_view"));
1922 popoutSplit->setWhatsThis(xi18nc("@info:whatsthis",
1923 "If the view has been split, this will pop the view in focus "
1924 "out into a new window."));
1925 popoutSplit->setIcon(QIcon::fromTheme(QStringLiteral("window-new")));
1926 actionCollection()->setDefaultShortcut(popoutSplit, Qt::SHIFT | Qt::Key_F3);
1927 connect(popoutSplit, &QAction::triggered, this, &DolphinMainWindow::popoutSplitView);
1928
1929 QAction *stashSplit = actionCollection()->addAction(QStringLiteral("split_stash"));
1930 actionCollection()->setDefaultShortcut(stashSplit, Qt::CTRL | Qt::Key_S);
1931 stashSplit->setText(i18nc("@action:intoolbar Stash", "Stash"));
1932 stashSplit->setToolTip(i18nc("@info", "Opens the stash virtual directory in a split window"));
1933 stashSplit->setIcon(QIcon::fromTheme(QStringLiteral("folder-stash")));
1934 stashSplit->setCheckable(false);
1935 QDBusConnectionInterface *sessionInterface = QDBusConnection::sessionBus().interface();
1936 stashSplit->setVisible(sessionInterface && sessionInterface->isServiceRegistered(QStringLiteral("org.kde.kio.StashNotifier")));
1937 connect(stashSplit, &QAction::triggered, this, &DolphinMainWindow::toggleSplitStash);
1938
1939 QAction *redisplay = KStandardAction::redisplay(this, &DolphinMainWindow::reloadView, actionCollection());
1940 redisplay->setToolTip(i18nc("@info:tooltip", "Refresh view"));
1941 redisplay->setWhatsThis(xi18nc("@info:whatsthis refresh",
1942 "<para>This refreshes "
1943 "the folder view.</para>"
1944 "<para>If the contents of this folder have changed, refreshing will re-scan this folder "
1945 "and show you a newly-updated view of the files and folders contained here.</para>"
1946 "<para>If the view is split, this refreshes the one that is currently in focus.</para>"));
1947
1948 QAction *stop = actionCollection()->addAction(QStringLiteral("stop"));
1949 stop->setText(i18nc("@action:inmenu View", "Stop"));
1950 stop->setToolTip(i18nc("@info", "Stop loading"));
1951 stop->setWhatsThis(i18nc("@info", "This stops the loading of the contents of the current folder."));
1952 stop->setIcon(QIcon::fromTheme(QStringLiteral("process-stop")));
1953 connect(stop, &QAction::triggered, this, &DolphinMainWindow::stopLoading);
1954
1955 KToggleAction *editableLocation = actionCollection()->add<KToggleAction>(QStringLiteral("editable_location"));
1956 editableLocation->setText(i18nc("@action:inmenu Navigation Bar", "Editable Location"));
1957 editableLocation->setWhatsThis(xi18nc("@info:whatsthis",
1958 "This toggles the <emphasis>Location Bar</emphasis> to be "
1959 "editable so you can directly enter a location you want to go to.<nl/>"
1960 "You can also switch to editing by clicking to the right of the "
1961 "location and switch back by confirming the edited location."));
1962 actionCollection()->setDefaultShortcut(editableLocation, Qt::Key_F6);
1963 connect(editableLocation, &KToggleAction::triggered, this, &DolphinMainWindow::toggleEditLocation);
1964
1965 QAction *replaceLocation = actionCollection()->addAction(QStringLiteral("replace_location"));
1966 replaceLocation->setText(i18nc("@action:inmenu Navigation Bar", "Replace Location"));
1967 // i18n: "enter" is used both in the meaning of "writing" and "going to" a new location here.
1968 // Both meanings are useful but not necessary to understand the use of "Replace Location".
1969 // So you might want to be more verbose in your language to convey the meaning but it's up to you.
1970 replaceLocation->setWhatsThis(xi18nc("@info:whatsthis",
1971 "This switches to editing the location and selects it "
1972 "so you can quickly enter a different location."));
1973 actionCollection()->setDefaultShortcuts(replaceLocation, {Qt::CTRL | Qt::Key_L, Qt::ALT | Qt::Key_D});
1974 connect(replaceLocation, &QAction::triggered, this, &DolphinMainWindow::replaceLocation);
1975
1976 // setup 'Go' menu
1977 {
1978 QScopedPointer<QAction> backAction(KStandardAction::back(nullptr, nullptr, nullptr));
1979 m_backAction = new KToolBarPopupAction(backAction->icon(), backAction->text(), actionCollection());
1980 m_backAction->setObjectName(backAction->objectName());
1981 m_backAction->setShortcuts(backAction->shortcuts());
1982 }
1983 m_backAction->setPopupMode(KToolBarPopupAction::DelayedPopup);
1984 connect(m_backAction, &QAction::triggered, this, &DolphinMainWindow::goBack);
1985 connect(m_backAction->popupMenu(), &QMenu::aboutToShow, this, &DolphinMainWindow::slotAboutToShowBackPopupMenu);
1986 connect(m_backAction->popupMenu(), &QMenu::triggered, this, &DolphinMainWindow::slotGoBack);
1987 actionCollection()->addAction(m_backAction->objectName(), m_backAction);
1988
1989 auto backShortcuts = m_backAction->shortcuts();
1990 // Prepend this shortcut, to avoid being hidden by the two-slot UI (#371130)
1991 backShortcuts.prepend(QKeySequence(Qt::Key_Backspace));
1992 actionCollection()->setDefaultShortcuts(m_backAction, backShortcuts);
1993
1994 DolphinRecentTabsMenu *recentTabsMenu = new DolphinRecentTabsMenu(this);
1995 actionCollection()->addAction(QStringLiteral("closed_tabs"), recentTabsMenu);
1996 connect(m_tabWidget, &DolphinTabWidget::rememberClosedTab, recentTabsMenu, &DolphinRecentTabsMenu::rememberClosedTab);
1997 connect(recentTabsMenu, &DolphinRecentTabsMenu::restoreClosedTab, m_tabWidget, &DolphinTabWidget::restoreClosedTab);
1998 connect(recentTabsMenu, &DolphinRecentTabsMenu::closedTabsCountChanged, this, &DolphinMainWindow::closedTabsCountChanged);
1999
2000 QAction *undoCloseTab = actionCollection()->addAction(QStringLiteral("undo_close_tab"));
2001 undoCloseTab->setText(i18nc("@action:inmenu File", "Undo close tab"));
2002 undoCloseTab->setWhatsThis(i18nc("@info:whatsthis undo close tab", "This returns you to the previously closed tab."));
2003 actionCollection()->setDefaultShortcut(undoCloseTab, Qt::CTRL | Qt::SHIFT | Qt::Key_T);
2004 undoCloseTab->setIcon(QIcon::fromTheme(QStringLiteral("edit-undo")));
2005 undoCloseTab->setEnabled(false);
2006 connect(undoCloseTab, &QAction::triggered, recentTabsMenu, &DolphinRecentTabsMenu::undoCloseTab);
2007
2008 auto undoAction = actionCollection()->action(KStandardAction::name(KStandardAction::Undo));
2009 undoAction->setWhatsThis(xi18nc("@info:whatsthis",
2010 "This undoes "
2011 "the last change you made to files or folders.<nl/>"
2012 "Such changes include <interface>creating</interface>, <interface>renaming</interface> "
2013 "and <interface>moving</interface> them to a different location "
2014 "or to the <filename>Trash</filename>. <nl/>Any changes that cannot be undone "
2015 "will ask for your confirmation beforehand."));
2016 undoAction->setEnabled(false); // undo should be disabled by default
2017
2018 {
2019 QScopedPointer<QAction> forwardAction(KStandardAction::forward(nullptr, nullptr, nullptr));
2020 m_forwardAction = new KToolBarPopupAction(forwardAction->icon(), forwardAction->text(), actionCollection());
2021 m_forwardAction->setObjectName(forwardAction->objectName());
2022 m_forwardAction->setShortcuts(forwardAction->shortcuts());
2023 }
2024 m_forwardAction->setPopupMode(KToolBarPopupAction::DelayedPopup);
2025 connect(m_forwardAction, &QAction::triggered, this, &DolphinMainWindow::goForward);
2026 connect(m_forwardAction->popupMenu(), &QMenu::aboutToShow, this, &DolphinMainWindow::slotAboutToShowForwardPopupMenu);
2027 connect(m_forwardAction->popupMenu(), &QMenu::triggered, this, &DolphinMainWindow::slotGoForward);
2028 actionCollection()->addAction(m_forwardAction->objectName(), m_forwardAction);
2029 actionCollection()->setDefaultShortcuts(m_forwardAction, m_forwardAction->shortcuts());
2030
2031 // enable middle-click to open in a new tab
2032 auto *middleClickEventFilter = new MiddleClickActionEventFilter(this);
2033 connect(middleClickEventFilter, &MiddleClickActionEventFilter::actionMiddleClicked, this, &DolphinMainWindow::slotBackForwardActionMiddleClicked);
2034 m_backAction->popupMenu()->installEventFilter(middleClickEventFilter);
2035 m_forwardAction->popupMenu()->installEventFilter(middleClickEventFilter);
2036 KStandardAction::up(this, &DolphinMainWindow::goUp, actionCollection());
2037 QAction *homeAction = KStandardAction::home(this, &DolphinMainWindow::goHome, actionCollection());
2038 homeAction->setWhatsThis(xi18nc("@info:whatsthis",
2039 "Go to your "
2040 "<filename>Home</filename> folder.<nl/>Every user account "
2041 "has their own <filename>Home</filename> that contains their personal files, "
2042 "as well as hidden folders for their applications' data and configuration files."));
2043
2044 // setup 'Tools' menu
2045 QAction *compareFiles = actionCollection()->addAction(QStringLiteral("compare_files"));
2046 compareFiles->setText(i18nc("@action:inmenu Tools", "Compare Files"));
2047 compareFiles->setIcon(QIcon::fromTheme(QStringLiteral("kompare")));
2048 compareFiles->setEnabled(false);
2049 connect(compareFiles, &QAction::triggered, this, &DolphinMainWindow::compareFiles);
2050
2051 QAction *openPreferredSearchTool = actionCollection()->addAction(QStringLiteral("open_preferred_search_tool"));
2052 openPreferredSearchTool->setText(i18nc("@action:inmenu Tools", "Open Preferred Search Tool"));
2053 openPreferredSearchTool->setWhatsThis(xi18nc("@info:whatsthis",
2054 "<para>This opens a preferred search tool for the viewed location.</para>"
2055 "<para>Use <emphasis>More Search Tools</emphasis> menu to configure it.</para>"));
2056 openPreferredSearchTool->setIcon(QIcon::fromTheme(QStringLiteral("search")));
2057 actionCollection()->setDefaultShortcut(openPreferredSearchTool, Qt::CTRL | Qt::SHIFT | Qt::Key_F);
2058 connect(openPreferredSearchTool, &QAction::triggered, this, &DolphinMainWindow::openPreferredSearchTool);
2059
2060 if (KAuthorized::authorize(QStringLiteral("shell_access"))) {
2061 QAction *openTerminal = actionCollection()->addAction(QStringLiteral("open_terminal"));
2062 openTerminal->setText(i18nc("@action:inmenu Tools", "Open Terminal"));
2063 openTerminal->setWhatsThis(xi18nc("@info:whatsthis",
2064 "<para>This opens a <emphasis>terminal</emphasis> application for the viewed location.</para>"
2065 "<para>To learn more about terminals use the help features in the terminal application.</para>"));
2066 openTerminal->setIcon(QIcon::fromTheme(QStringLiteral("utilities-terminal")));
2067 actionCollection()->setDefaultShortcut(openTerminal, Qt::SHIFT | Qt::Key_F4);
2068 connect(openTerminal, &QAction::triggered, this, &DolphinMainWindow::openTerminal);
2069
2070 QAction *openTerminalHere = actionCollection()->addAction(QStringLiteral("open_terminal_here"));
2071 // i18n: "Here" refers to the location(s) of the currently selected item(s) or the currently viewed location if nothing is selected.
2072 openTerminalHere->setText(i18nc("@action:inmenu Tools", "Open Terminal Here"));
2073 openTerminalHere->setWhatsThis(xi18nc("@info:whatsthis",
2074 "<para>This opens <emphasis>terminal</emphasis> applications for the selected items' locations.</para>"
2075 "<para>To learn more about terminals use the help features in the terminal application.</para>"));
2076 openTerminalHere->setIcon(QIcon::fromTheme(QStringLiteral("utilities-terminal")));
2077 actionCollection()->setDefaultShortcut(openTerminalHere, Qt::SHIFT | Qt::ALT | Qt::Key_F4);
2078 connect(openTerminalHere, &QAction::triggered, this, &DolphinMainWindow::openTerminalHere);
2079 }
2080
2081 // setup 'Bookmarks' menu
2082 KActionMenu *bookmarkMenu = new KActionMenu(i18nc("@title:menu", "&Bookmarks"), this);
2083 bookmarkMenu->setIcon(QIcon::fromTheme(QStringLiteral("bookmarks")));
2084 // Make the toolbar button version work properly on click
2085 bookmarkMenu->setPopupMode(QToolButton::InstantPopup);
2086 m_bookmarkHandler = new DolphinBookmarkHandler(this, actionCollection(), bookmarkMenu->menu(), this);
2087 actionCollection()->addAction(QStringLiteral("bookmarks"), bookmarkMenu);
2088
2089 // setup 'Settings' menu
2090 KToggleAction *showMenuBar = KStandardAction::showMenubar(nullptr, nullptr, actionCollection());
2091 showMenuBar->setWhatsThis(xi18nc("@info:whatsthis",
2092 "<para>This switches between having a <emphasis>Menubar</emphasis> "
2093 "and having an <interface>%1</interface> button. Both "
2094 "contain mostly the same actions and configuration options.</para>"
2095 "<para>The Menubar takes up more space but allows for fast and organized access to all "
2096 "actions an application has to offer.</para><para>The %1 button "
2097 "is simpler and small which makes triggering advanced actions more time consuming.</para>",
2098 hamburgerMenuAction->text().replace('&', "")));
2099 connect(showMenuBar,
2100 &KToggleAction::triggered, // Fixes #286822
2101 this,
2102 &DolphinMainWindow::toggleShowMenuBar,
2103 Qt::QueuedConnection);
2104
2105 KToggleAction *showStatusBar = KStandardAction::showStatusbar(nullptr, nullptr, actionCollection());
2106 showStatusBar->setChecked(GeneralSettings::showStatusBar());
2107 connect(GeneralSettings::self(), &GeneralSettings::showStatusBarChanged, showStatusBar, &KToggleAction::setChecked);
2108 connect(showStatusBar, &KToggleAction::triggered, this, [this](bool checked) {
2109 GeneralSettings::setShowStatusBar(checked);
2110 refreshViews();
2111 });
2112
2113 KStandardAction::keyBindings(this, &DolphinMainWindow::slotKeyBindings, actionCollection());
2114 KStandardAction::preferences(this, &DolphinMainWindow::editSettings, actionCollection());
2115
2116 // not in menu actions
2117 QList<QKeySequence> nextTabKeys = KStandardShortcut::tabNext();
2118 nextTabKeys.append(QKeySequence(Qt::CTRL | Qt::Key_Tab));
2119
2120 QList<QKeySequence> prevTabKeys = KStandardShortcut::tabPrev();
2121 prevTabKeys.append(QKeySequence(Qt::CTRL | Qt::SHIFT | Qt::Key_Tab));
2122
2123 for (int i = 0; i < MaxActivateTabShortcuts; ++i) {
2124 QAction *activateTab = actionCollection()->addAction(QStringLiteral("activate_tab_%1").arg(i));
2125 activateTab->setText(i18nc("@action:inmenu", "Go to Tab %1", i + 1));
2126 activateTab->setEnabled(false);
2127 connect(activateTab, &QAction::triggered, this, [this, i]() {
2128 m_tabWidget->activateTab(i);
2129 });
2130
2131 // only add default shortcuts for the first 9 tabs regardless of MaxActivateTabShortcuts
2132 if (i < 9) {
2133 actionCollection()->setDefaultShortcut(activateTab, QStringLiteral("Alt+%1").arg(i + 1));
2134 }
2135 }
2136
2137 QAction *activateLastTab = actionCollection()->addAction(QStringLiteral("activate_last_tab"));
2138 activateLastTab->setIconText(i18nc("@action:inmenu", "Last Tab"));
2139 activateLastTab->setText(i18nc("@action:inmenu", "Go to Last Tab"));
2140 activateLastTab->setEnabled(false);
2141 connect(activateLastTab, &QAction::triggered, m_tabWidget, &DolphinTabWidget::activateLastTab);
2142 actionCollection()->setDefaultShortcut(activateLastTab, Qt::ALT | Qt::Key_0);
2143
2144 QAction *activateNextTab = actionCollection()->addAction(QStringLiteral("activate_next_tab"));
2145 activateNextTab->setIconText(i18nc("@action:inmenu", "Next Tab"));
2146 activateNextTab->setText(i18nc("@action:inmenu", "Go to Next Tab"));
2147 activateNextTab->setEnabled(false);
2148 connect(activateNextTab, &QAction::triggered, m_tabWidget, &DolphinTabWidget::activateNextTab);
2149 actionCollection()->setDefaultShortcuts(activateNextTab, nextTabKeys);
2150
2151 QAction *activatePrevTab = actionCollection()->addAction(QStringLiteral("activate_prev_tab"));
2152 activatePrevTab->setIconText(i18nc("@action:inmenu", "Previous Tab"));
2153 activatePrevTab->setText(i18nc("@action:inmenu", "Go to Previous Tab"));
2154 activatePrevTab->setEnabled(false);
2155 connect(activatePrevTab, &QAction::triggered, m_tabWidget, &DolphinTabWidget::activatePrevTab);
2156 actionCollection()->setDefaultShortcuts(activatePrevTab, prevTabKeys);
2157
2158 // for context menu
2159 QAction *showTarget = actionCollection()->addAction(QStringLiteral("show_target"));
2160 showTarget->setText(i18nc("@action:inmenu", "Show Target"));
2161 showTarget->setIcon(QIcon::fromTheme(QStringLiteral("document-open-folder")));
2162 showTarget->setEnabled(false);
2163 connect(showTarget, &QAction::triggered, this, &DolphinMainWindow::showTarget);
2164
2165 QAction *openInNewTab = actionCollection()->addAction(QStringLiteral("open_in_new_tab"));
2166 openInNewTab->setText(i18nc("@action:inmenu", "Open in New Tab"));
2167 openInNewTab->setIcon(QIcon::fromTheme(QStringLiteral("tab-new")));
2168 connect(openInNewTab, &QAction::triggered, this, &DolphinMainWindow::openInNewTab);
2169
2170 QAction *openInNewTabs = actionCollection()->addAction(QStringLiteral("open_in_new_tabs"));
2171 openInNewTabs->setText(i18nc("@action:inmenu", "Open in New Tabs"));
2172 openInNewTabs->setIcon(QIcon::fromTheme(QStringLiteral("tab-new")));
2173 connect(openInNewTabs, &QAction::triggered, this, &DolphinMainWindow::openInNewTab);
2174
2175 QAction *openInNewWindow = actionCollection()->addAction(QStringLiteral("open_in_new_window"));
2176 openInNewWindow->setText(i18nc("@action:inmenu", "Open in New Window"));
2177 openInNewWindow->setIcon(QIcon::fromTheme(QStringLiteral("window-new")));
2178 connect(openInNewWindow, &QAction::triggered, this, &DolphinMainWindow::openInNewWindow);
2179
2180 QAction *openInSplitViewAction = actionCollection()->addAction(QStringLiteral("open_in_split_view"));
2181 openInSplitViewAction->setText(i18nc("@action:inmenu", "Open in Split View"));
2182 openInSplitViewAction->setIcon(QIcon::fromTheme(QStringLiteral("view-right-new")));
2183 connect(openInSplitViewAction, &QAction::triggered, this, [this]() {
2184 openInSplitView(QUrl());
2185 });
2186
2187 m_recentFiles = new KRecentFilesAction(this);
2188 }
2189
2190 void DolphinMainWindow::setupDockWidgets()
2191 {
2192 const bool lock = GeneralSettings::lockPanels();
2193
2194 DolphinPlacesModelSingleton::instance().placesModel()->setPanelsLocked(lock);
2195
2196 KDualAction *lockLayoutAction = actionCollection()->add<KDualAction>(QStringLiteral("lock_panels"));
2197 lockLayoutAction->setActiveText(i18nc("@action:inmenu Panels", "Unlock Panels"));
2198 lockLayoutAction->setActiveIcon(QIcon::fromTheme(QStringLiteral("object-unlocked")));
2199 lockLayoutAction->setInactiveText(i18nc("@action:inmenu Panels", "Lock Panels"));
2200 lockLayoutAction->setInactiveIcon(QIcon::fromTheme(QStringLiteral("object-locked")));
2201 lockLayoutAction->setWhatsThis(xi18nc("@info:whatsthis",
2202 "This "
2203 "switches between having panels <emphasis>locked</emphasis> or "
2204 "<emphasis>unlocked</emphasis>.<nl/>Unlocked panels can be "
2205 "dragged to the other side of the window and have a close "
2206 "button.<nl/>Locked panels are embedded more cleanly."));
2207 lockLayoutAction->setActive(lock);
2208 connect(lockLayoutAction, &KDualAction::triggered, this, &DolphinMainWindow::togglePanelLockState);
2209
2210 // Setup "Information"
2211 DolphinDockWidget *infoDock = new DolphinDockWidget(i18nc("@title:window", "Information"));
2212 infoDock->setLocked(lock);
2213 infoDock->setObjectName(QStringLiteral("infoDock"));
2214 infoDock->setAllowedAreas(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea);
2215
2216 #if HAVE_BALOO
2217 InformationPanel *infoPanel = new InformationPanel(infoDock);
2218 infoPanel->setCustomContextMenuActions({lockLayoutAction});
2219 connect(infoPanel, &InformationPanel::urlActivated, this, &DolphinMainWindow::handleUrl);
2220 infoDock->setWidget(infoPanel);
2221
2222 createPanelAction(QIcon::fromTheme(QStringLiteral("documentinfo")), Qt::Key_F11, infoDock, QStringLiteral("show_information_panel"));
2223
2224 addDockWidget(Qt::RightDockWidgetArea, infoDock);
2225 connect(this, &DolphinMainWindow::urlChanged, infoPanel, &InformationPanel::setUrl);
2226 connect(this, &DolphinMainWindow::selectionChanged, infoPanel, &InformationPanel::setSelection);
2227 connect(this, &DolphinMainWindow::requestItemInfo, infoPanel, &InformationPanel::requestDelayedItemInfo);
2228 connect(this, &DolphinMainWindow::fileItemsChanged, infoPanel, &InformationPanel::slotFilesItemChanged);
2229 connect(this, &DolphinMainWindow::settingsChanged, infoPanel, &InformationPanel::readSettings);
2230 #endif
2231
2232 // i18n: This is the last paragraph for the "What's This"-texts of all four panels.
2233 const QString panelWhatsThis = xi18nc("@info:whatsthis",
2234 "<para>To show or "
2235 "hide panels like this go to <interface>Menu|Panels</interface> "
2236 "or <interface>View|Panels</interface>.</para>");
2237 #if HAVE_BALOO
2238 actionCollection()
2239 ->action(QStringLiteral("show_information_panel"))
2240 ->setWhatsThis(xi18nc("@info:whatsthis",
2241 "<para> This toggles the "
2242 "<emphasis>information</emphasis> panel at the right side of the "
2243 "window.</para><para>The panel provides in-depth information "
2244 "about the items your mouse is hovering over or about the selected "
2245 "items. Otherwise it informs you about the currently viewed folder.<nl/>"
2246 "For single items a preview of their contents is provided.</para>"));
2247 #endif
2248 infoDock->setWhatsThis(xi18nc("@info:whatsthis",
2249 "<para>This panel "
2250 "provides in-depth information about the items your mouse is "
2251 "hovering over or about the selected items. Otherwise it informs "
2252 "you about the currently viewed folder.<nl/>For single items a "
2253 "preview of their contents is provided.</para><para>You can configure "
2254 "which and how details are given here by right-clicking.</para>")
2255 + panelWhatsThis);
2256
2257 // Setup "Folders"
2258 DolphinDockWidget *foldersDock = new DolphinDockWidget(i18nc("@title:window", "Folders"));
2259 foldersDock->setLocked(lock);
2260 foldersDock->setObjectName(QStringLiteral("foldersDock"));
2261 foldersDock->setAllowedAreas(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea);
2262 FoldersPanel *foldersPanel = new FoldersPanel(foldersDock);
2263 foldersPanel->setCustomContextMenuActions({lockLayoutAction});
2264 foldersDock->setWidget(foldersPanel);
2265
2266 createPanelAction(QIcon::fromTheme(QStringLiteral("folder")), Qt::Key_F7, foldersDock, QStringLiteral("show_folders_panel"));
2267
2268 addDockWidget(Qt::LeftDockWidgetArea, foldersDock);
2269 connect(this, &DolphinMainWindow::urlChanged, foldersPanel, &FoldersPanel::setUrl);
2270 connect(foldersPanel, &FoldersPanel::folderActivated, this, &DolphinMainWindow::changeUrl);
2271 connect(foldersPanel, &FoldersPanel::folderInNewTab, this, &DolphinMainWindow::openNewTab);
2272 connect(foldersPanel, &FoldersPanel::folderInNewActiveTab, this, &DolphinMainWindow::openNewTabAndActivate);
2273 connect(foldersPanel, &FoldersPanel::errorMessage, this, &DolphinMainWindow::showErrorMessage);
2274
2275 actionCollection()
2276 ->action(QStringLiteral("show_folders_panel"))
2277 ->setWhatsThis(xi18nc("@info:whatsthis",
2278 "This toggles the "
2279 "<emphasis>folders</emphasis> panel at the left side of the window."
2280 "<nl/><nl/>It shows the folders of the <emphasis>file system"
2281 "</emphasis> in a <emphasis>tree view</emphasis>."));
2282 foldersDock->setWhatsThis(xi18nc("@info:whatsthis",
2283 "<para>This panel "
2284 "shows the folders of the <emphasis>file system</emphasis> in a "
2285 "<emphasis>tree view</emphasis>.</para><para>Click a folder to go "
2286 "there. Click the arrow to the left of a folder to see its subfolders. "
2287 "This allows quick switching between any folders.</para>")
2288 + panelWhatsThis);
2289
2290 // Setup "Terminal"
2291 #if HAVE_TERMINAL
2292 if (KAuthorized::authorize(QStringLiteral("shell_access"))) {
2293 DolphinDockWidget *terminalDock = new DolphinDockWidget(i18nc("@title:window Shell terminal", "Terminal"));
2294 terminalDock->setLocked(lock);
2295 terminalDock->setObjectName(QStringLiteral("terminalDock"));
2296 terminalDock->setContentsMargins(0, 0, 0, 0);
2297 m_terminalPanel = new TerminalPanel(terminalDock);
2298 m_terminalPanel->setCustomContextMenuActions({lockLayoutAction});
2299 terminalDock->setWidget(m_terminalPanel);
2300
2301 connect(m_terminalPanel, &TerminalPanel::hideTerminalPanel, terminalDock, &DolphinDockWidget::hide);
2302 connect(m_terminalPanel, &TerminalPanel::changeUrl, this, &DolphinMainWindow::slotTerminalDirectoryChanged);
2303 connect(terminalDock, &DolphinDockWidget::visibilityChanged, m_terminalPanel, &TerminalPanel::dockVisibilityChanged);
2304 connect(terminalDock, &DolphinDockWidget::visibilityChanged, this, &DolphinMainWindow::slotTerminalPanelVisibilityChanged);
2305
2306 createPanelAction(QIcon::fromTheme(QStringLiteral("dialog-scripts")), Qt::Key_F4, terminalDock, QStringLiteral("show_terminal_panel"));
2307
2308 addDockWidget(Qt::BottomDockWidgetArea, terminalDock);
2309 connect(this, &DolphinMainWindow::urlChanged, m_terminalPanel, &TerminalPanel::setUrl);
2310
2311 if (GeneralSettings::version() < 200) {
2312 terminalDock->hide();
2313 }
2314
2315 actionCollection()
2316 ->action(QStringLiteral("show_terminal_panel"))
2317 ->setWhatsThis(xi18nc("@info:whatsthis",
2318 "<para>This toggles the "
2319 "<emphasis>terminal</emphasis> panel at the bottom of the window."
2320 "<nl/>The location in the terminal will always match the folder "
2321 "view so you can navigate using either.</para><para>The terminal "
2322 "panel is not needed for basic computer usage but can be useful "
2323 "for advanced tasks. To learn more about terminals use the help features "
2324 "in a standalone terminal application like Konsole.</para>"));
2325 terminalDock->setWhatsThis(xi18nc("@info:whatsthis",
2326 "<para>This is "
2327 "the <emphasis>terminal</emphasis> panel. It behaves like a "
2328 "normal terminal but will match the location of the folder view "
2329 "so you can navigate using either.</para><para>The terminal panel "
2330 "is not needed for basic computer usage but can be useful for "
2331 "advanced tasks. To learn more about terminals use the help features in a "
2332 "standalone terminal application like Konsole.</para>")
2333 + panelWhatsThis);
2334
2335 QAction *focusTerminalPanel = actionCollection()->addAction(QStringLiteral("focus_terminal_panel"));
2336 focusTerminalPanel->setText(i18nc("@action:inmenu Tools", "Focus Terminal Panel"));
2337 focusTerminalPanel->setToolTip(i18nc("@info:tooltip", "Move keyboard focus to and from the Terminal panel."));
2338 focusTerminalPanel->setIcon(QIcon::fromTheme(QStringLiteral("swap-panels")));
2339 actionCollection()->setDefaultShortcut(focusTerminalPanel, Qt::CTRL | Qt::SHIFT | Qt::Key_F4);
2340 connect(focusTerminalPanel, &QAction::triggered, this, &DolphinMainWindow::toggleTerminalPanelFocus);
2341 } // endif "shell_access" allowed
2342 #endif // HAVE_TERMINAL
2343
2344 if (GeneralSettings::version() < 200) {
2345 infoDock->hide();
2346 foldersDock->hide();
2347 }
2348
2349 // Setup "Places"
2350 DolphinDockWidget *placesDock = new DolphinDockWidget(i18nc("@title:window", "Places"));
2351 placesDock->setLocked(lock);
2352 placesDock->setObjectName(QStringLiteral("placesDock"));
2353 placesDock->setAllowedAreas(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea);
2354
2355 m_placesPanel = new PlacesPanel(placesDock);
2356 m_placesPanel->setCustomContextMenuActions({lockLayoutAction});
2357 placesDock->setWidget(m_placesPanel);
2358
2359 createPanelAction(QIcon::fromTheme(QStringLiteral("compass")), Qt::Key_F9, placesDock, QStringLiteral("show_places_panel"));
2360
2361 addDockWidget(Qt::LeftDockWidgetArea, placesDock);
2362 connect(m_placesPanel, &PlacesPanel::placeActivated, this, &DolphinMainWindow::slotPlaceActivated);
2363 connect(m_placesPanel, &PlacesPanel::tabRequested, this, &DolphinMainWindow::openNewTab);
2364 connect(m_placesPanel, &PlacesPanel::activeTabRequested, this, &DolphinMainWindow::openNewTabAndActivate);
2365 connect(m_placesPanel, &PlacesPanel::newWindowRequested, this, [this](const QUrl &url) {
2366 Dolphin::openNewWindow({url}, this);
2367 });
2368 connect(m_placesPanel, &PlacesPanel::openInSplitViewRequested, this, &DolphinMainWindow::openInSplitView);
2369 connect(m_placesPanel, &PlacesPanel::errorMessage, this, &DolphinMainWindow::showErrorMessage);
2370 connect(this, &DolphinMainWindow::urlChanged, m_placesPanel, &PlacesPanel::setUrl);
2371 connect(placesDock, &DolphinDockWidget::visibilityChanged, &DolphinUrlNavigatorsController::slotPlacesPanelVisibilityChanged);
2372 connect(placesDock, &DolphinDockWidget::visibilityChanged, this, &DolphinMainWindow::slotPlacesPanelVisibilityChanged);
2373 connect(this, &DolphinMainWindow::settingsChanged, m_placesPanel, &PlacesPanel::readSettings);
2374 connect(m_placesPanel, &PlacesPanel::storageTearDownRequested, this, &DolphinMainWindow::slotStorageTearDownFromPlacesRequested);
2375 connect(m_placesPanel, &PlacesPanel::storageTearDownExternallyRequested, this, &DolphinMainWindow::slotStorageTearDownExternallyRequested);
2376 DolphinUrlNavigatorsController::slotPlacesPanelVisibilityChanged(m_placesPanel->isVisible());
2377
2378 auto actionShowAllPlaces = new QAction(QIcon::fromTheme(QStringLiteral("view-hidden")), i18nc("@item:inmenu", "Show Hidden Places"), this);
2379 actionShowAllPlaces->setCheckable(true);
2380 actionShowAllPlaces->setDisabled(true);
2381 actionShowAllPlaces->setWhatsThis(i18nc("@info:whatsthis",
2382 "This displays "
2383 "all places in the places panel that have been hidden. They will "
2384 "appear semi-transparent and allow you to uncheck their \"Hide\" property."));
2385
2386 connect(actionShowAllPlaces, &QAction::triggered, this, [this](bool checked) {
2387 m_placesPanel->setShowAll(checked);
2388 });
2389 connect(m_placesPanel, &PlacesPanel::allPlacesShownChanged, actionShowAllPlaces, &QAction::setChecked);
2390
2391 actionCollection()
2392 ->action(QStringLiteral("show_places_panel"))
2393 ->setWhatsThis(xi18nc("@info:whatsthis",
2394 "<para>This toggles the "
2395 "<emphasis>places</emphasis> panel at the left side of the window."
2396 "</para><para>It allows you to go to locations you have "
2397 "bookmarked and to access disk or media attached to the computer "
2398 "or to the network. It also contains sections to find recently "
2399 "saved files or files of a certain type.</para>"));
2400 placesDock->setWhatsThis(xi18nc("@info:whatsthis",
2401 "<para>This is the "
2402 "<emphasis>Places</emphasis> panel. It allows you to go to locations "
2403 "you have bookmarked and to access disk or media attached to the "
2404 "computer or to the network. It also contains sections to find "
2405 "recently saved files or files of a certain type.</para><para>"
2406 "Click on an entry to go there. Click with the right mouse button "
2407 "instead to open any entry in a new tab or new window.</para>"
2408 "<para>New entries can be added by dragging folders onto this panel. "
2409 "Right-click any section or entry to hide it. Right-click an empty "
2410 "space on this panel and select <interface>Show Hidden Places"
2411 "</interface> to display it again.</para>")
2412 + panelWhatsThis);
2413
2414 QAction *focusPlacesPanel = actionCollection()->addAction(QStringLiteral("focus_places_panel"));
2415 focusPlacesPanel->setText(i18nc("@action:inmenu View", "Focus Places Panel"));
2416 focusPlacesPanel->setToolTip(i18nc("@info:tooltip", "Move keyboard focus to and from the Places panel."));
2417 focusPlacesPanel->setIcon(QIcon::fromTheme(QStringLiteral("swap-panels")));
2418 actionCollection()->setDefaultShortcut(focusPlacesPanel, Qt::CTRL | Qt::Key_P);
2419 connect(focusPlacesPanel, &QAction::triggered, this, &DolphinMainWindow::togglePlacesPanelFocus);
2420
2421 // Add actions into the "Panels" menu
2422 KActionMenu *panelsMenu = new KActionMenu(i18nc("@action:inmenu View", "Show Panels"), this);
2423 actionCollection()->addAction(QStringLiteral("panels"), panelsMenu);
2424 panelsMenu->setIcon(QIcon::fromTheme(QStringLiteral("view-sidetree")));
2425 panelsMenu->setPopupMode(QToolButton::InstantPopup);
2426 const KActionCollection *ac = actionCollection();
2427 panelsMenu->addAction(ac->action(QStringLiteral("show_places_panel")));
2428 #if HAVE_BALOO
2429 panelsMenu->addAction(ac->action(QStringLiteral("show_information_panel")));
2430 #endif
2431 panelsMenu->addAction(ac->action(QStringLiteral("show_folders_panel")));
2432 panelsMenu->addAction(ac->action(QStringLiteral("show_terminal_panel")));
2433 panelsMenu->addSeparator();
2434 panelsMenu->addAction(lockLayoutAction);
2435 panelsMenu->addSeparator();
2436 panelsMenu->addAction(actionShowAllPlaces);
2437 panelsMenu->addAction(focusPlacesPanel);
2438 panelsMenu->addAction(ac->action(QStringLiteral("focus_terminal_panel")));
2439
2440 connect(panelsMenu->menu(), &QMenu::aboutToShow, this, [actionShowAllPlaces] {
2441 actionShowAllPlaces->setEnabled(DolphinPlacesModelSingleton::instance().placesModel()->hiddenCount());
2442 });
2443 }
2444
2445 void DolphinMainWindow::updateFileAndEditActions()
2446 {
2447 const KFileItemList list = m_activeViewContainer->view()->selectedItems();
2448 const KActionCollection *col = actionCollection();
2449 KFileItemListProperties capabilitiesSource(list);
2450
2451 QAction *renameAction = col->action(KStandardAction::name(KStandardAction::RenameFile));
2452 QAction *moveToTrashAction = col->action(KStandardAction::name(KStandardAction::MoveToTrash));
2453 QAction *deleteAction = col->action(KStandardAction::name(KStandardAction::DeleteFile));
2454 QAction *cutAction = col->action(KStandardAction::name(KStandardAction::Cut));
2455 QAction *duplicateAction = col->action(QStringLiteral("duplicate")); // see DolphinViewActionHandler
2456 QAction *addToPlacesAction = col->action(QStringLiteral("add_to_places"));
2457 QAction *copyToOtherViewAction = col->action(QStringLiteral("copy_to_inactive_split_view"));
2458 QAction *moveToOtherViewAction = col->action(QStringLiteral("move_to_inactive_split_view"));
2459 QAction *copyLocation = col->action(QStringLiteral("copy_location"));
2460
2461 if (list.isEmpty()) {
2462 stateChanged(QStringLiteral("has_no_selection"));
2463
2464 // All actions that need a selection to function can be enabled because they should trigger selection mode.
2465 renameAction->setEnabled(true);
2466 moveToTrashAction->setEnabled(true);
2467 deleteAction->setEnabled(true);
2468 cutAction->setEnabled(true);
2469 duplicateAction->setEnabled(true);
2470 addToPlacesAction->setEnabled(true);
2471 copyLocation->setEnabled(true);
2472 // Them triggering selection mode and not directly acting on selected items is signified by adding "…" to their text.
2473 m_actionTextHelper->textsWhenNothingIsSelectedEnabled(true);
2474
2475 } else {
2476 m_actionTextHelper->textsWhenNothingIsSelectedEnabled(false);
2477 stateChanged(QStringLiteral("has_selection"));
2478
2479 QAction *deleteWithTrashShortcut = col->action(QStringLiteral("delete_shortcut")); // see DolphinViewActionHandler
2480 QAction *showTarget = col->action(QStringLiteral("show_target"));
2481
2482 if (list.length() == 1 && list.first().isDir()) {
2483 addToPlacesAction->setEnabled(true);
2484 } else {
2485 addToPlacesAction->setEnabled(false);
2486 }
2487
2488 const bool enableMoveToTrash = capabilitiesSource.isLocal() && capabilitiesSource.supportsMoving();
2489
2490 renameAction->setEnabled(capabilitiesSource.supportsMoving());
2491 m_disabledActionNotifier->setDisabledReason(renameAction, i18nc("@info", "Cannot rename: You do not have permission to rename items in this folder."));
2492 deleteAction->setEnabled(capabilitiesSource.supportsDeleting());
2493 m_disabledActionNotifier->setDisabledReason(deleteAction,
2494 i18nc("@info", "Cannot delete: You do not have permission to remove items from this folder."));
2495 cutAction->setEnabled(capabilitiesSource.supportsMoving());
2496 m_disabledActionNotifier->setDisabledReason(cutAction, i18nc("@info", "Cannot cut: You do not have permission to move items from this folder."));
2497 copyLocation->setEnabled(list.length() == 1);
2498 showTarget->setEnabled(list.length() == 1 && list.at(0).isLink());
2499 duplicateAction->setEnabled(capabilitiesSource.supportsWriting());
2500 m_disabledActionNotifier->setDisabledReason(duplicateAction,
2501 i18nc("@info", "Cannot duplicate here: You do not have permission to create items in this folder."));
2502
2503 if (enableMoveToTrash) {
2504 moveToTrashAction->setEnabled(true);
2505 deleteWithTrashShortcut->setEnabled(false);
2506 m_disabledActionNotifier->clearDisabledReason(deleteWithTrashShortcut);
2507 } else {
2508 moveToTrashAction->setEnabled(false);
2509 deleteWithTrashShortcut->setEnabled(capabilitiesSource.supportsDeleting());
2510 m_disabledActionNotifier->setDisabledReason(deleteWithTrashShortcut,
2511 i18nc("@info", "Cannot delete: You do not have permission to remove items from this folder."));
2512 }
2513 }
2514
2515 if (!m_tabWidget->currentTabPage()->splitViewEnabled()) {
2516 // No need to set the disabled reason here, as it's obvious to the user that the reason is the split view being disabled.
2517 copyToOtherViewAction->setEnabled(false);
2518 m_disabledActionNotifier->clearDisabledReason(copyToOtherViewAction);
2519 moveToOtherViewAction->setEnabled(false);
2520 m_disabledActionNotifier->clearDisabledReason(moveToOtherViewAction);
2521 } else if (list.isEmpty()) {
2522 copyToOtherViewAction->setEnabled(false);
2523 m_disabledActionNotifier->setDisabledReason(copyToOtherViewAction, i18nc("@info", "Cannot copy to other view: No files selected."));
2524 moveToOtherViewAction->setEnabled(false);
2525 m_disabledActionNotifier->setDisabledReason(moveToOtherViewAction, i18nc("@info", "Cannot move to other view: No files selected."));
2526 } else {
2527 DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
2528 KFileItem capabilitiesDestination;
2529
2530 if (tabPage->primaryViewActive()) {
2531 capabilitiesDestination = tabPage->secondaryViewContainer()->rootItem();
2532 } else {
2533 capabilitiesDestination = tabPage->primaryViewContainer()->rootItem();
2534 }
2535
2536 const auto destUrl = capabilitiesDestination.url();
2537 const bool allNotTargetOrigin = std::all_of(list.cbegin(), list.cend(), [destUrl](const KFileItem &item) {
2538 return item.url().adjusted(QUrl::RemoveFilename | QUrl::StripTrailingSlash) != destUrl;
2539 });
2540
2541 if (!allNotTargetOrigin) {
2542 copyToOtherViewAction->setEnabled(false);
2543 m_disabledActionNotifier->setDisabledReason(copyToOtherViewAction,
2544 i18nc("@info", "Cannot copy to other view: The other view already contains these items."));
2545 moveToOtherViewAction->setEnabled(false);
2546 m_disabledActionNotifier->setDisabledReason(moveToOtherViewAction,
2547 i18nc("@info", "Cannot move to other view: The other view already contains these items."));
2548 } else if (!capabilitiesDestination.isWritable()) {
2549 copyToOtherViewAction->setEnabled(false);
2550 m_disabledActionNotifier->setDisabledReason(
2551 copyToOtherViewAction,
2552 i18nc("@info", "Cannot copy to other view: You do not have permission to write into the destination folder."));
2553 moveToOtherViewAction->setEnabled(false);
2554 m_disabledActionNotifier->setDisabledReason(
2555 moveToOtherViewAction,
2556 i18nc("@info", "Cannot move to other view: You do not have permission to write into the destination folder."));
2557 } else {
2558 copyToOtherViewAction->setEnabled(true);
2559 moveToOtherViewAction->setEnabled(capabilitiesSource.supportsMoving());
2560 m_disabledActionNotifier->setDisabledReason(
2561 moveToOtherViewAction,
2562 i18nc("@info", "Cannot move to other view: You do not have permission to move items from this folder."));
2563 }
2564 }
2565 }
2566
2567 void DolphinMainWindow::updateViewActions()
2568 {
2569 m_actionHandler->updateViewActions();
2570
2571 QAction *toggleFilterBarAction = actionCollection()->action(QStringLiteral("toggle_filter"));
2572 toggleFilterBarAction->setChecked(m_activeViewContainer->isFilterBarVisible());
2573
2574 updateSplitActions();
2575 }
2576
2577 void DolphinMainWindow::updateGoActions()
2578 {
2579 QAction *goUpAction = actionCollection()->action(KStandardAction::name(KStandardAction::Up));
2580 const QUrl currentUrl = m_activeViewContainer->url();
2581 // I think this is one of the best places to firstly be confronted
2582 // with a file system and its hierarchy. Talking about the root
2583 // directory might seem too much here but it is the question that
2584 // naturally arises in this context.
2585 goUpAction->setWhatsThis(xi18nc("@info:whatsthis",
2586 "<para>Go to "
2587 "the folder that contains the currently viewed one.</para>"
2588 "<para>All files and folders are organized in a hierarchical "
2589 "<emphasis>file system</emphasis>. At the top of this hierarchy is "
2590 "a directory that contains all data connected to this computer"
2591 "—the <emphasis>root directory</emphasis>.</para>"));
2592 goUpAction->setEnabled(KIO::upUrl(currentUrl) != currentUrl);
2593 }
2594
2595 void DolphinMainWindow::refreshViews()
2596 {
2597 m_tabWidget->refreshViews();
2598
2599 if (GeneralSettings::modifiedStartupSettings()) {
2600 updateWindowTitle();
2601 }
2602
2603 updateSplitActions();
2604
2605 Q_EMIT settingsChanged();
2606 }
2607
2608 void DolphinMainWindow::clearStatusBar()
2609 {
2610 m_activeViewContainer->statusBar()->resetToDefaultText();
2611 }
2612
2613 void DolphinMainWindow::connectViewSignals(DolphinViewContainer *container)
2614 {
2615 connect(container, &DolphinViewContainer::showFilterBarChanged, this, &DolphinMainWindow::updateFilterBarAction);
2616 connect(container, &DolphinViewContainer::writeStateChanged, this, &DolphinMainWindow::slotWriteStateChanged);
2617 slotWriteStateChanged(container->view()->isFolderWritable());
2618 connect(container, &DolphinViewContainer::searchModeEnabledChanged, this, &DolphinMainWindow::updateSearchAction);
2619 connect(container, &DolphinViewContainer::captionChanged, this, &DolphinMainWindow::updateWindowTitle);
2620 connect(container, &DolphinViewContainer::tabRequested, this, &DolphinMainWindow::openNewTab);
2621 connect(container, &DolphinViewContainer::activeTabRequested, this, &DolphinMainWindow::openNewTabAndActivate);
2622
2623 const QAction *toggleSearchAction = actionCollection()->action(QStringLiteral("toggle_search"));
2624 connect(toggleSearchAction, &QAction::triggered, container, &DolphinViewContainer::setSearchModeEnabled);
2625
2626 // Make the toggled state of the selection mode actions visually follow the selection mode state of the view.
2627 auto toggleSelectionModeAction = actionCollection()->action(QStringLiteral("toggle_selection_mode"));
2628 toggleSelectionModeAction->setChecked(m_activeViewContainer->isSelectionModeEnabled());
2629 connect(m_activeViewContainer, &DolphinViewContainer::selectionModeChanged, toggleSelectionModeAction, &QAction::setChecked);
2630
2631 const DolphinView *view = container->view();
2632 connect(view, &DolphinView::selectionChanged, this, &DolphinMainWindow::slotSelectionChanged);
2633 connect(view, &DolphinView::requestItemInfo, this, &DolphinMainWindow::requestItemInfo);
2634 connect(view, &DolphinView::fileItemsChanged, this, &DolphinMainWindow::fileItemsChanged);
2635 connect(view, &DolphinView::tabRequested, this, &DolphinMainWindow::openNewTab);
2636 connect(view, &DolphinView::activeTabRequested, this, &DolphinMainWindow::openNewTabAndActivate);
2637 connect(view, &DolphinView::windowRequested, this, &DolphinMainWindow::openNewWindow);
2638 connect(view, &DolphinView::requestContextMenu, this, &DolphinMainWindow::openContextMenu);
2639 connect(view, &DolphinView::directoryLoadingStarted, this, &DolphinMainWindow::enableStopAction);
2640 connect(view, &DolphinView::directoryLoadingCompleted, this, &DolphinMainWindow::disableStopAction);
2641 connect(view, &DolphinView::directoryLoadingCompleted, this, &DolphinMainWindow::slotDirectoryLoadingCompleted);
2642 connect(view, &DolphinView::goBackRequested, this, &DolphinMainWindow::goBack);
2643 connect(view, &DolphinView::goForwardRequested, this, &DolphinMainWindow::goForward);
2644 connect(view, &DolphinView::urlActivated, this, &DolphinMainWindow::handleUrl);
2645 connect(view, &DolphinView::goUpRequested, this, &DolphinMainWindow::goUp);
2646 connect(view, &DolphinView::doubleClickViewBackground, this, &DolphinMainWindow::slotDoubleClickViewBackground);
2647
2648 connect(container->urlNavigatorInternalWithHistory(), &KUrlNavigator::urlChanged, this, &DolphinMainWindow::changeUrl);
2649 connect(container->urlNavigatorInternalWithHistory(), &KUrlNavigator::historyChanged, this, &DolphinMainWindow::updateHistory);
2650
2651 auto navigators = static_cast<DolphinNavigatorsWidgetAction *>(actionCollection()->action(QStringLiteral("url_navigators")));
2652 const KUrlNavigator *navigator =
2653 m_tabWidget->currentTabPage()->primaryViewActive() ? navigators->primaryUrlNavigator() : navigators->secondaryUrlNavigator();
2654
2655 QAction *editableLocactionAction = actionCollection()->action(QStringLiteral("editable_location"));
2656 editableLocactionAction->setChecked(navigator->isUrlEditable());
2657 connect(navigator, &KUrlNavigator::editableStateChanged, this, &DolphinMainWindow::slotEditableStateChanged);
2658 connect(navigator, &KUrlNavigator::tabRequested, this, &DolphinMainWindow::openNewTab);
2659 connect(navigator, &KUrlNavigator::activeTabRequested, this, &DolphinMainWindow::openNewTabAndActivate);
2660 connect(navigator, &KUrlNavigator::newWindowRequested, this, &DolphinMainWindow::openNewWindow);
2661 }
2662
2663 void DolphinMainWindow::updateSplitActions()
2664 {
2665 QAction *popoutSplitAction = actionCollection()->action(QStringLiteral("popout_split_view"));
2666
2667 auto setActionPopupMode = [this](KActionMenu *action, QToolButton::ToolButtonPopupMode popupMode) {
2668 action->setPopupMode(popupMode);
2669 if (auto *buttonForAction = qobject_cast<QToolButton *>(toolBar()->widgetForAction(action))) {
2670 buttonForAction->setPopupMode(popupMode);
2671 }
2672 };
2673
2674 const DolphinTabPage *tabPage = m_tabWidget->currentTabPage();
2675 if (tabPage->splitViewEnabled()) {
2676 if (GeneralSettings::closeActiveSplitView() ? tabPage->primaryViewActive() : !tabPage->primaryViewActive()) {
2677 m_splitViewAction->setText(i18nc("@action:intoolbar Close left view", "Close"));
2678 m_splitViewAction->setToolTip(i18nc("@info", "Close left view"));
2679 m_splitViewAction->setIcon(QIcon::fromTheme(QStringLiteral("view-left-close")));
2680 popoutSplitAction->setText(i18nc("@action:intoolbar Move left view to a new window", "Pop out Left View"));
2681 popoutSplitAction->setToolTip(i18nc("@info", "Move left view to a new window"));
2682 } else {
2683 m_splitViewAction->setText(i18nc("@action:intoolbar Close right view", "Close"));
2684 m_splitViewAction->setToolTip(i18nc("@info", "Close right view"));
2685 m_splitViewAction->setIcon(QIcon::fromTheme(QStringLiteral("view-right-close")));
2686 popoutSplitAction->setText(i18nc("@action:intoolbar Move right view to a new window", "Pop out Right View"));
2687 popoutSplitAction->setToolTip(i18nc("@info", "Move right view to a new window"));
2688 }
2689 popoutSplitAction->setEnabled(true);
2690 if (!m_splitViewAction->menu()) {
2691 setActionPopupMode(m_splitViewAction, QToolButton::MenuButtonPopup);
2692 m_splitViewAction->setMenu(new QMenu);
2693 m_splitViewAction->addAction(popoutSplitAction);
2694 }
2695 } else {
2696 m_splitViewAction->setText(i18nc("@action:intoolbar Split view", "Split"));
2697 m_splitViewAction->setToolTip(i18nc("@info", "Split view"));
2698 m_splitViewAction->setIcon(QIcon::fromTheme(QStringLiteral("view-right-new")));
2699 popoutSplitAction->setText(i18nc("@action:intoolbar Move view in focus to a new window", "Pop out"));
2700 popoutSplitAction->setEnabled(false);
2701 if (m_splitViewAction->menu()) {
2702 m_splitViewAction->removeAction(popoutSplitAction);
2703 m_splitViewAction->menu()->deleteLater();
2704 m_splitViewAction->setMenu(nullptr);
2705 setActionPopupMode(m_splitViewAction, QToolButton::DelayedPopup);
2706 }
2707 }
2708
2709 // Update state from toolbar action
2710 m_splitViewMenuAction->setText(m_splitViewAction->text());
2711 m_splitViewMenuAction->setToolTip(m_splitViewAction->toolTip());
2712 m_splitViewMenuAction->setIcon(m_splitViewAction->icon());
2713 }
2714
2715 void DolphinMainWindow::updateAllowedToolbarAreas()
2716 {
2717 auto navigators = static_cast<DolphinNavigatorsWidgetAction *>(actionCollection()->action(QStringLiteral("url_navigators")));
2718 if (toolBar()->actions().contains(navigators)) {
2719 toolBar()->setAllowedAreas(Qt::TopToolBarArea | Qt::BottomToolBarArea);
2720 if (toolBarArea(toolBar()) == Qt::LeftToolBarArea || toolBarArea(toolBar()) == Qt::RightToolBarArea) {
2721 addToolBar(Qt::TopToolBarArea, toolBar());
2722 }
2723 } else {
2724 toolBar()->setAllowedAreas(Qt::AllToolBarAreas);
2725 }
2726 }
2727
2728 bool DolphinMainWindow::isKompareInstalled() const
2729 {
2730 static bool initialized = false;
2731 static bool installed = false;
2732 if (!initialized) {
2733 // TODO: maybe replace this approach later by using a menu
2734 // plugin like kdiff3plugin.cpp
2735 installed = !QStandardPaths::findExecutable(QStringLiteral("kompare")).isEmpty();
2736 initialized = true;
2737 }
2738 return installed;
2739 }
2740
2741 void DolphinMainWindow::createPanelAction(const QIcon &icon, const QKeySequence &shortcut, QDockWidget *dockWidget, const QString &actionName)
2742 {
2743 auto dockAction = dockWidget->toggleViewAction();
2744 dockAction->setIcon(icon);
2745 dockAction->setEnabled(true);
2746
2747 QAction *panelAction = actionCollection()->addAction(actionName, dockAction);
2748 actionCollection()->setDefaultShortcut(panelAction, shortcut);
2749 }
2750 // clang-format off
2751 void DolphinMainWindow::setupWhatsThis()
2752 {
2753 // main widgets
2754 menuBar()->setWhatsThis(xi18nc("@info:whatsthis", "<para>This is the "
2755 "<emphasis>Menubar</emphasis>. It provides access to commands and "
2756 "configuration options. Left-click on any of the menus on this "
2757 "bar to see its contents.</para><para>The Menubar can be hidden "
2758 "by unchecking <interface>Settings|Show Menubar</interface>. Then "
2759 "most of its contents become available through a <interface>Menu"
2760 "</interface> button on the <emphasis>Toolbar</emphasis>.</para>"));
2761 toolBar()->setWhatsThis(xi18nc("@info:whatsthis", "<para>This is the "
2762 "<emphasis>Toolbar</emphasis>. It allows quick access to "
2763 "frequently used actions.</para><para>It is highly customizable. "
2764 "All items you see in the <interface>Menu</interface> or "
2765 "in the <interface>Menubar</interface> can be placed on the "
2766 "Toolbar. Just right-click on it and select <interface>Configure "
2767 "Toolbars…</interface> or find this action within the <interface>"
2768 "menu</interface>."
2769 "</para><para>The location of the bar and the style of its "
2770 "buttons can also be changed in the right-click menu. Right-click "
2771 "a button if you want to show or hide its text.</para>"));
2772 m_tabWidget->setWhatsThis(xi18nc("@info:whatsthis main view",
2773 "<para>Here you can see the <emphasis>folders</emphasis> and "
2774 "<emphasis>files</emphasis> that are at the location described in "
2775 "the <interface>Location Bar</interface> above. This area is the "
2776 "central part of this application where you navigate to the files "
2777 "you want to use.</para><para>For an elaborate and general "
2778 "introduction to this application <link "
2779 "url='https://userbase.kde.org/Dolphin/File_Management#Introduction_to_Dolphin'>"
2780 "click here</link>. This will open an introductory article from "
2781 "the <emphasis>KDE UserBase Wiki</emphasis>.</para><para>For brief "
2782 "explanations of all the features of this <emphasis>view</emphasis> "
2783 "<link url='help:/dolphin/dolphin-view.html'>click here</link> "
2784 "instead. This will open a page from the <emphasis>Handbook"
2785 "</emphasis> that covers the basics.</para>"));
2786
2787 // Settings menu
2788 actionCollection()->action(KStandardAction::name(KStandardAction::KeyBindings))
2789 ->setWhatsThis(xi18nc("@info:whatsthis","<para>This opens a window "
2790 "that lists the <emphasis>keyboard shortcuts</emphasis>.<nl/>"
2791 "There you can set up key combinations to trigger an action when "
2792 "they are pressed simultaneously. All commands in this application can "
2793 "be triggered this way.</para>"));
2794 actionCollection()->action(KStandardAction::name(KStandardAction::ConfigureToolbars))
2795 ->setWhatsThis(xi18nc("@info:whatsthis","<para>This opens a window in which "
2796 "you can change which buttons appear on the <emphasis>Toolbar</emphasis>.</para>"
2797 "<para>All items you see in the <interface>Menu</interface> can also be placed on the Toolbar.</para>"));
2798 actionCollection()->action(KStandardAction::name(KStandardAction::Preferences))
2799 ->setWhatsThis(xi18nc("@info:whatsthis","This opens a window where you can "
2800 "change a multitude of settings for this application. For an explanation "
2801 "of the various settings go to the chapter <emphasis>Configuring Dolphin"
2802 "</emphasis> in <interface>Help|Dolphin Handbook</interface>."));
2803
2804 // Help menu
2805
2806 auto setStandardActionWhatsThis = [this](KStandardAction::StandardAction actionId,
2807 const QString &whatsThis) {
2808 // Check for the existence of an action since it can be restricted through the Kiosk system
2809 if (auto *action = actionCollection()->action(KStandardAction::name(actionId))) {
2810 action->setWhatsThis(whatsThis);
2811 }
2812 };
2813
2814 // i18n: If the external link isn't available in your language it might make
2815 // sense to state the external link's language in brackets to not
2816 // frustrate the user. If there are multiple languages that the user might
2817 // know with a reasonable chance you might want to have 2 external links.
2818 // The same might be true for any external link you translate.
2819 setStandardActionWhatsThis(KStandardAction::HelpContents, xi18nc("@info:whatsthis handbook", "<para>This opens the Handbook for this application. It provides explanations for every part of <emphasis>Dolphin</emphasis>.</para><para>If you want more elaborate introductions to the different features of <emphasis>Dolphin</emphasis> <link url='https://userbase.kde.org/Dolphin/File_Management'>click here</link>. It will open the dedicated page in the KDE UserBase Wiki.</para>"));
2820 // (The i18n call should be completely in the line following the i18n: comment without any line breaks within the i18n call or the comment might not be correctly extracted. See: https://commits.kde.org/kxmlgui/a31135046e1b3335b5d7bbbe6aa9a883ce3284c1 )
2821
2822 setStandardActionWhatsThis(KStandardAction::WhatsThis,
2823 xi18nc("@info:whatsthis whatsthis button",
2824 "<para>This is the button that invokes the help feature you are "
2825 "using right now! Click it, then click any component of this "
2826 "application to ask \"What's this?\" about it. The mouse cursor "
2827 "will change appearance if no help is available for a spot.</para>"
2828 "<para>There are two other ways to get help: "
2829 "The <link url='help:/dolphin/index.html'>Dolphin Handbook</link> and "
2830 "the <link url='https://userbase.kde.org/Dolphin/File_Management'>KDE "
2831 "UserBase Wiki</link>.</para><para>The \"What's this?\" help is "
2832 "missing in most other windows so don't get too used to this.</para>"));
2833
2834 setStandardActionWhatsThis(KStandardAction::ReportBug,
2835 xi18nc("@info:whatsthis","<para>This opens a "
2836 "window that will guide you through reporting errors or flaws "
2837 "in this application or in other KDE software.</para>"
2838 "<para>High-quality bug reports are much appreciated. To learn "
2839 "how to make your bug report as effective as possible "
2840 "<link url='https://community.kde.org/Get_Involved/Bug_Reporting'>"
2841 "click here</link>.</para>"));
2842
2843 setStandardActionWhatsThis(KStandardAction::Donate,
2844 xi18nc("@info:whatsthis", "<para>This opens a "
2845 "<emphasis>web page</emphasis> where you can donate to "
2846 "support the continued work on this application and many "
2847 "other projects by the <emphasis>KDE</emphasis> community.</para>"
2848 "<para>Donating is the easiest and fastest way to efficiently "
2849 "support KDE and its projects. KDE projects are available for "
2850 "free therefore your donation is needed to cover things that "
2851 "require money like servers, contributor meetings, etc.</para>"
2852 "<para><emphasis>KDE e.V.</emphasis> is the non-profit "
2853 "organization behind the KDE community.</para>"));
2854
2855 setStandardActionWhatsThis(KStandardAction::SwitchApplicationLanguage,
2856 xi18nc("@info:whatsthis",
2857 "With this you can change the language this application uses."
2858 "<nl/>You can even set secondary languages which will be used "
2859 "if texts are not available in your preferred language."));
2860
2861 setStandardActionWhatsThis(KStandardAction::AboutApp,
2862 xi18nc("@info:whatsthis","This opens a "
2863 "window that informs you about the version, license, "
2864 "used libraries and maintainers of this application."));
2865
2866 setStandardActionWhatsThis(KStandardAction::AboutKDE,
2867 xi18nc("@info:whatsthis","This opens a "
2868 "window with information about <emphasis>KDE</emphasis>. "
2869 "The KDE community are the people behind this free software."
2870 "<nl/>If you like using this application but don't know "
2871 "about KDE or want to see a cute dragon have a look!"));
2872 }
2873 // clang-format on
2874
2875 bool DolphinMainWindow::addHamburgerMenuToToolbar()
2876 {
2877 QDomDocument domDocument = KXMLGUIClient::domDocument();
2878 if (domDocument.isNull()) {
2879 return false;
2880 }
2881 QDomNode toolbar = domDocument.elementsByTagName(QStringLiteral("ToolBar")).at(0);
2882 if (toolbar.isNull()) {
2883 return false;
2884 }
2885
2886 QDomElement hamburgerMenuElement = domDocument.createElement(QStringLiteral("Action"));
2887 hamburgerMenuElement.setAttribute(QStringLiteral("name"), QStringLiteral("hamburger_menu"));
2888 toolbar.appendChild(hamburgerMenuElement);
2889
2890 KXMLGUIFactory::saveConfigFile(domDocument, xmlFile());
2891 reloadXML();
2892 createGUI();
2893 return true;
2894 // Make sure to also remove the <KXMLGUIFactory> and <QDomDocument> include
2895 // whenever this method is removed (maybe in the year ~2026).
2896 }
2897
2898 // Set a sane initial window size
2899 QSize DolphinMainWindow::sizeHint() const
2900 {
2901 return KXmlGuiWindow::sizeHint().expandedTo(QSize(760, 550));
2902 }
2903
2904 void DolphinMainWindow::saveNewToolbarConfig()
2905 {
2906 KXmlGuiWindow::saveNewToolbarConfig(); // Applies the new config. This has to be called first
2907 // because the rest of this method decides things
2908 // based on the new config.
2909 auto navigators = static_cast<DolphinNavigatorsWidgetAction *>(actionCollection()->action(QStringLiteral("url_navigators")));
2910 if (!toolBar()->actions().contains(navigators)) {
2911 m_tabWidget->currentTabPage()->insertNavigatorsWidget(navigators);
2912 }
2913 updateAllowedToolbarAreas();
2914 (static_cast<KHamburgerMenu *>(actionCollection()->action(KStandardAction::name(KStandardAction::HamburgerMenu))))->hideActionsOf(toolBar());
2915 }
2916
2917 void DolphinMainWindow::toggleTerminalPanelFocus()
2918 {
2919 if (!m_terminalPanel->isVisible()) {
2920 actionCollection()->action(QStringLiteral("show_terminal_panel"))->trigger(); // Also moves focus to the panel.
2921 actionCollection()->action(QStringLiteral("focus_terminal_panel"))->setText(i18nc("@action:inmenu Tools", "Defocus Terminal Panel"));
2922 return;
2923 }
2924
2925 if (m_terminalPanel->terminalHasFocus()) {
2926 m_activeViewContainer->view()->setFocus(Qt::FocusReason::ShortcutFocusReason);
2927 actionCollection()->action(QStringLiteral("focus_terminal_panel"))->setText(i18nc("@action:inmenu Tools", "Focus Terminal Panel"));
2928 return;
2929 }
2930
2931 m_terminalPanel->setFocus(Qt::FocusReason::ShortcutFocusReason);
2932 actionCollection()->action(QStringLiteral("focus_terminal_panel"))->setText(i18nc("@action:inmenu Tools", "Defocus Terminal Panel"));
2933 }
2934
2935 void DolphinMainWindow::togglePlacesPanelFocus()
2936 {
2937 if (!m_placesPanel->isVisible()) {
2938 actionCollection()->action(QStringLiteral("show_places_panel"))->trigger(); // Also moves focus to the panel.
2939 actionCollection()->action(QStringLiteral("focus_places_panel"))->setText(i18nc("@action:inmenu View", "Defocus Terminal Panel"));
2940 return;
2941 }
2942
2943 if (m_placesPanel->hasFocus()) {
2944 m_activeViewContainer->view()->setFocus(Qt::FocusReason::ShortcutFocusReason);
2945 actionCollection()->action(QStringLiteral("focus_places_panel"))->setText(i18nc("@action:inmenu View", "Focus Places Panel"));
2946 return;
2947 }
2948
2949 m_placesPanel->setFocus(Qt::FocusReason::ShortcutFocusReason);
2950 actionCollection()->action(QStringLiteral("focus_places_panel"))->setText(i18nc("@action:inmenu View", "Defocus Places Panel"));
2951 }
2952
2953 DolphinMainWindow::UndoUiInterface::UndoUiInterface()
2954 : KIO::FileUndoManager::UiInterface()
2955 {
2956 }
2957
2958 DolphinMainWindow::UndoUiInterface::~UndoUiInterface()
2959 {
2960 }
2961
2962 void DolphinMainWindow::UndoUiInterface::jobError(KIO::Job *job)
2963 {
2964 DolphinMainWindow *mainWin = qobject_cast<DolphinMainWindow *>(parentWidget());
2965 if (mainWin) {
2966 DolphinViewContainer *container = mainWin->activeViewContainer();
2967 container->showMessage(job->errorString(), KMessageWidget::Error);
2968 } else {
2969 KIO::FileUndoManager::UiInterface::jobError(job);
2970 }
2971 }
2972
2973 bool DolphinMainWindow::isUrlOpen(const QString &url)
2974 {
2975 return m_tabWidget->isUrlOpen(QUrl::fromUserInput(url));
2976 }
2977
2978 bool DolphinMainWindow::isItemVisibleInAnyView(const QString &urlOfItem)
2979 {
2980 return m_tabWidget->isItemVisibleInAnyView(QUrl::fromUserInput(urlOfItem));
2981 }
2982
2983 void DolphinMainWindow::slotDoubleClickViewBackground(Qt::MouseButton button)
2984 {
2985 if (button != Qt::MouseButton::LeftButton) {
2986 // only handle left mouse button for now
2987 return;
2988 }
2989
2990 GeneralSettings *settings = GeneralSettings::self();
2991 QString clickAction = settings->doubleClickViewAction();
2992
2993 DolphinView *view = activeViewContainer()->view();
2994 if (view == nullptr || clickAction == "none") {
2995 return;
2996 }
2997
2998 if (clickAction == customCommand) {
2999 // run custom command set by the user
3000 QString path = view->url().toLocalFile();
3001 QString clickCustomAction = settings->doubleClickViewCustomAction();
3002 clickCustomAction.replace("{path}", path.prepend('"').append('"'));
3003
3004 m_job = new KIO::CommandLauncherJob(clickCustomAction);
3005 m_job->setUiDelegate(new KDialogJobUiDelegate(KJobUiDelegate::AutoHandlingEnabled, this));
3006 m_job->start();
3007
3008 } else {
3009 // get the action set by the user and trigger it
3010 const KActionCollection *actions = actionCollection();
3011 QAction *action = actions->action(clickAction);
3012 if (action == nullptr) {
3013 qCWarning(DolphinDebug) << QStringLiteral("Double-click view: action `%1` was not found").arg(clickAction);
3014 return;
3015 }
3016 action->trigger();
3017 }
3018 }
3019
3020 #include "moc_dolphinmainwindow.cpp"