2 * SPDX-FileCopyrightText: 2006 Peter Penz (peter.penz@gmx.at) and Cvetoslav Ludmiloff
4 * SPDX-License-Identifier: GPL-2.0-or-later
7 #include "dolphincontextmenu.h"
9 #include "dolphin_contextmenusettings.h"
10 #include "dolphin_generalsettings.h"
11 #include "dolphinmainwindow.h"
12 #include "dolphinnewfilemenu.h"
13 #include "dolphinplacesmodelsingleton.h"
14 #include "dolphinremoveaction.h"
15 #include "dolphinviewcontainer.h"
17 #include "trash/dolphintrash.h"
18 #include "views/dolphinview.h"
19 #include "views/viewmodecontroller.h"
21 #include <KActionCollection>
22 #include <KFileItemListProperties>
23 #include <KHamburgerMenu>
24 #include <KIO/EmptyTrashJob>
25 #include <KIO/JobUiDelegate>
27 #include <KIO/RestoreJob>
28 #include <KJobWidgets>
29 #include <KLocalizedString>
30 #include <KNewFileMenu>
31 #include <KStandardAction>
32 #include <kio_version.h>
34 #include <QApplication>
38 DolphinContextMenu::DolphinContextMenu(DolphinMainWindow
*parent
,
39 const KFileItem
&fileInfo
,
40 const KFileItemList
&selectedItems
,
42 KFileItemActions
*fileItemActions
)
44 , m_mainWindow(parent
)
45 , m_fileInfo(fileInfo
)
47 , m_baseFileItem(nullptr)
48 , m_selectedItems(selectedItems
)
49 , m_selectedItemsProperties(nullptr)
50 , m_context(NoContext
)
51 , m_copyToMenu(parent
)
52 , m_removeAction(nullptr)
53 , m_fileItemActions(fileItemActions
)
55 QApplication::instance()->installEventFilter(this);
60 DolphinContextMenu::~DolphinContextMenu()
62 delete m_baseFileItem
;
63 m_baseFileItem
= nullptr;
64 delete m_selectedItemsProperties
;
65 m_selectedItemsProperties
= nullptr;
68 void DolphinContextMenu::addAllActions()
70 static_cast<KHamburgerMenu
*>(m_mainWindow
->actionCollection()->action(QStringLiteral("hamburger_menu")))->addToMenu(this);
72 // get the context information
73 const auto scheme
= m_baseUrl
.scheme();
74 if (scheme
== QLatin1String("trash")) {
75 m_context
|= TrashContext
;
76 } else if (scheme
.contains(QLatin1String("search"))) {
77 m_context
|= SearchContext
;
78 } else if (scheme
.contains(QLatin1String("timeline"))) {
79 m_context
|= TimelineContext
;
82 if (!m_fileInfo
.isNull() && !m_selectedItems
.isEmpty()) {
83 m_context
|= ItemContext
;
84 // TODO: handle other use cases like devices + desktop files
87 // open the corresponding popup for the context
88 if (m_context
& TrashContext
) {
89 if (m_context
& ItemContext
) {
90 addTrashItemContextMenu();
92 addTrashContextMenu();
94 } else if (m_context
& ItemContext
) {
97 addViewportContextMenu();
101 bool DolphinContextMenu::eventFilter(QObject
*object
, QEvent
*event
)
105 if (event
->type() == QEvent::KeyPress
|| event
->type() == QEvent::KeyRelease
) {
106 QKeyEvent
*keyEvent
= static_cast<QKeyEvent
*>(event
);
108 if (m_removeAction
&& keyEvent
->key() == Qt::Key_Shift
) {
109 if (event
->type() == QEvent::KeyPress
) {
110 m_removeAction
->update(DolphinRemoveAction::ShiftState::Pressed
);
112 m_removeAction
->update(DolphinRemoveAction::ShiftState::Released
);
120 void DolphinContextMenu::addTrashContextMenu()
122 Q_ASSERT(m_context
& TrashContext
);
124 QAction
*emptyTrashAction
= addAction(QIcon::fromTheme(QStringLiteral("trash-empty")), i18nc("@action:inmenu", "Empty Trash"), [this]() {
125 Trash::empty(m_mainWindow
);
127 emptyTrashAction
->setEnabled(!Trash::isEmpty());
129 QAction
*propertiesAction
= m_mainWindow
->actionCollection()->action(QStringLiteral("properties"));
130 addAction(propertiesAction
);
133 void DolphinContextMenu::addTrashItemContextMenu()
135 Q_ASSERT(m_context
& TrashContext
);
136 Q_ASSERT(m_context
& ItemContext
);
138 addAction(QIcon::fromTheme("restoration"), i18nc("@action:inmenu", "Restore"), [this]() {
139 QList
<QUrl
> selectedUrls
;
140 selectedUrls
.reserve(m_selectedItems
.count());
141 for (const KFileItem
&item
: qAsConst(m_selectedItems
)) {
142 selectedUrls
.append(item
.url());
145 KIO::RestoreJob
*job
= KIO::restoreFromTrash(selectedUrls
);
146 KJobWidgets::setWindow(job
, m_mainWindow
);
147 job
->uiDelegate()->setAutoErrorHandlingEnabled(true);
150 QAction
*deleteAction
= m_mainWindow
->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile
));
151 addAction(deleteAction
);
153 QAction
*propertiesAction
= m_mainWindow
->actionCollection()->action(QStringLiteral("properties"));
154 addAction(propertiesAction
);
157 void DolphinContextMenu::addDirectoryItemContextMenu()
159 // insert 'Open in new window' and 'Open in new tab' entries
160 const KFileItemListProperties
&selectedItemsProps
= selectedItemsProperties();
161 if (ContextMenuSettings::showOpenInNewTab()) {
162 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("open_in_new_tab")));
164 if (ContextMenuSettings::showOpenInNewWindow()) {
165 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("open_in_new_window")));
168 // Insert 'Open With' entries
169 addOpenWithActions();
171 // set up 'Create New' menu
172 DolphinNewFileMenu
*newFileMenu
= new DolphinNewFileMenu(m_mainWindow
->actionCollection(), m_mainWindow
);
173 #if KIO_VERSION >= QT_VERSION_CHECK(5, 100, 0)
174 newFileMenu
->setNewFolderShortcutAction(m_mainWindow
->actionCollection()->action("create_dir"));
176 newFileMenu
->checkUpToDate();
177 #if KIO_VERSION >= QT_VERSION_CHECK(5, 97, 0)
178 newFileMenu
->setWorkingDirectory(m_fileInfo
.url());
180 newFileMenu
->setPopupFiles(QList
<QUrl
>() << m_fileInfo
.url());
182 newFileMenu
->setEnabled(selectedItemsProps
.supportsWriting());
183 connect(newFileMenu
, &DolphinNewFileMenu::fileCreated
, newFileMenu
, &DolphinNewFileMenu::deleteLater
);
184 connect(newFileMenu
, &DolphinNewFileMenu::directoryCreated
, newFileMenu
, &DolphinNewFileMenu::deleteLater
);
186 QMenu
*menu
= newFileMenu
->menu();
187 menu
->setTitle(i18nc("@title:menu Create new folder, file, link, etc.", "Create New"));
188 menu
->setIcon(QIcon::fromTheme(QStringLiteral("list-add")));
194 void DolphinContextMenu::addItemContextMenu()
196 Q_ASSERT(!m_fileInfo
.isNull());
198 const KFileItemListProperties
&selectedItemsProps
= selectedItemsProperties();
200 m_fileItemActions
->setItemListProperties(selectedItemsProps
);
202 if (m_selectedItems
.count() == 1) {
204 if (m_fileInfo
.isDir()) {
205 addDirectoryItemContextMenu();
206 } else if (m_context
& TimelineContext
|| m_context
& SearchContext
) {
207 addOpenWithActions();
209 addAction(QIcon::fromTheme(QStringLiteral("document-open-folder")), i18nc("@action:inmenu", "Open Path"), [this]() {
210 m_mainWindow
->changeUrl(KIO::upUrl(m_fileInfo
.url()));
211 m_mainWindow
->activeViewContainer()->view()->markUrlsAsSelected({m_fileInfo
.url()});
212 m_mainWindow
->activeViewContainer()->view()->markUrlAsCurrent(m_fileInfo
.url());
215 addAction(QIcon::fromTheme(QStringLiteral("window-new")), i18nc("@action:inmenu", "Open Path in New Window"), [this]() {
216 Dolphin::openNewWindow({m_fileInfo
.url()}, m_mainWindow
, Dolphin::OpenNewWindowFlag::Select
);
219 addAction(QIcon::fromTheme(QStringLiteral("tab-new")), i18nc("@action:inmenu", "Open Path in New Tab"), [this]() {
220 m_mainWindow
->openNewTab(KIO::upUrl(m_fileInfo
.url()));
225 // Insert 'Open With" entries
226 addOpenWithActions();
228 if (m_fileInfo
.isLink()) {
229 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("show_target")));
234 bool selectionHasOnlyDirs
= true;
235 for (const auto &item
: qAsConst(m_selectedItems
)) {
236 const QUrl
&url
= DolphinView::openItemAsFolderUrl(item
);
238 selectionHasOnlyDirs
= false;
243 if (selectionHasOnlyDirs
&& ContextMenuSettings::showOpenInNewTab()) {
244 // insert 'Open in new tab' entry
245 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("open_in_new_tabs")));
247 // Insert 'Open With" entries
248 addOpenWithActions();
251 insertDefaultItemActions(selectedItemsProps
);
253 addAdditionalActions(selectedItemsProps
);
255 // insert 'Copy To' and 'Move To' sub menus
256 if (ContextMenuSettings::showCopyMoveMenu()) {
257 m_copyToMenu
.setUrls(m_selectedItems
.urlList());
258 m_copyToMenu
.setReadOnly(!selectedItemsProps
.supportsWriting());
259 m_copyToMenu
.setAutoErrorHandlingEnabled(true);
260 m_copyToMenu
.addActionsTo(this);
263 if (m_mainWindow
->isSplitViewEnabledInCurrentTab()) {
264 if (ContextMenuSettings::showCopyToOtherSplitView()) {
265 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("copy_to_inactive_split_view")));
268 if (ContextMenuSettings::showMoveToOtherSplitView()) {
269 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("move_to_inactive_split_view")));
273 // insert 'Properties...' entry
275 QAction
*propertiesAction
= m_mainWindow
->actionCollection()->action(QStringLiteral("properties"));
276 addAction(propertiesAction
);
279 void DolphinContextMenu::addViewportContextMenu()
281 const KFileItemListProperties
baseUrlProperties(KFileItemList() << baseFileItem());
282 m_fileItemActions
->setItemListProperties(baseUrlProperties
);
284 // Set up and insert 'Create New' menu
285 KNewFileMenu
*newFileMenu
= m_mainWindow
->newFileMenu();
286 newFileMenu
->checkUpToDate();
287 #if KIO_VERSION >= QT_VERSION_CHECK(5, 97, 0)
288 newFileMenu
->setWorkingDirectory(m_baseUrl
);
290 newFileMenu
->setPopupFiles(QList
<QUrl
>() << m_baseUrl
);
292 addMenu(newFileMenu
->menu());
294 // Show "open with" menu items even if the dir is empty, because there are legitimate
295 // use cases for this, such as opening an empty dir in Kate or VSCode or something
296 addOpenWithActions();
298 QAction
*pasteAction
= createPasteAction();
300 addAction(pasteAction
);
303 // Insert 'Add to Places' entry if it's not already in the places panel
304 if (ContextMenuSettings::showAddToPlaces() && !placeExists(m_mainWindow
->activeViewContainer()->url())) {
305 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("add_to_places")));
309 // Insert 'Sort By' and 'View Mode'
310 if (ContextMenuSettings::showSortBy()) {
311 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("sort")));
313 if (ContextMenuSettings::showViewMode()) {
314 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("view_mode")));
316 if (ContextMenuSettings::showSortBy() || ContextMenuSettings::showViewMode()) {
320 addAdditionalActions(baseUrlProperties
);
324 QAction
*propertiesAction
= m_mainWindow
->actionCollection()->action(QStringLiteral("properties"));
325 addAction(propertiesAction
);
328 void DolphinContextMenu::insertDefaultItemActions(const KFileItemListProperties
&properties
)
330 const KActionCollection
*collection
= m_mainWindow
->actionCollection();
332 // Insert 'Cut', 'Copy', 'Copy Location' and 'Paste'
333 addAction(collection
->action(KStandardAction::name(KStandardAction::Cut
)));
334 addAction(collection
->action(KStandardAction::name(KStandardAction::Copy
)));
335 if (ContextMenuSettings::showCopyLocation()) {
336 QAction
*copyPathAction
= collection
->action(QString("copy_location"));
337 copyPathAction
->setEnabled(m_selectedItems
.size() == 1);
338 addAction(copyPathAction
);
340 QAction
*pasteAction
= createPasteAction();
342 addAction(pasteAction
);
345 // Insert 'Duplicate Here'
346 if (ContextMenuSettings::showDuplicateHere()) {
347 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("duplicate")));
351 addAction(collection
->action(KStandardAction::name(KStandardAction::RenameFile
)));
353 // Insert 'Add to Places' entry if appropriate
354 if (ContextMenuSettings::showAddToPlaces() && m_selectedItems
.count() == 1 && m_fileInfo
.isDir() && !placeExists(m_fileInfo
.url())) {
355 addAction(m_mainWindow
->actionCollection()->action(QStringLiteral("add_to_places")));
360 // Insert 'Move to Trash' and/or 'Delete'
361 const bool showDeleteAction
= (KSharedConfig::openConfig()->group("KDE").readEntry("ShowDeleteCommand", false) || !properties
.isLocal());
362 const bool showMoveToTrashAction
= (properties
.isLocal() && properties
.supportsMoving());
364 if (showDeleteAction
&& showMoveToTrashAction
) {
365 delete m_removeAction
;
366 m_removeAction
= nullptr;
367 addAction(m_mainWindow
->actionCollection()->action(KStandardAction::name(KStandardAction::MoveToTrash
)));
368 addAction(m_mainWindow
->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile
)));
369 } else if (showDeleteAction
&& !showMoveToTrashAction
) {
370 addAction(m_mainWindow
->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile
)));
372 if (!m_removeAction
) {
373 m_removeAction
= new DolphinRemoveAction(this, m_mainWindow
->actionCollection());
375 addAction(m_removeAction
);
376 m_removeAction
->update();
380 bool DolphinContextMenu::placeExists(const QUrl
&url
) const
382 const KFilePlacesModel
*placesModel
= DolphinPlacesModelSingleton::instance().placesModel();
384 const auto &matchedPlaces
= placesModel
->match(placesModel
->index(0, 0), KFilePlacesModel::UrlRole
, url
, 1, Qt::MatchExactly
);
386 return !matchedPlaces
.isEmpty();
389 QAction
*DolphinContextMenu::createPasteAction()
391 QAction
*action
= nullptr;
393 if (!m_fileInfo
.isNull() && m_selectedItems
.count() <= 1) {
394 destItem
= m_fileInfo
;
396 destItem
= baseFileItem();
399 if (!destItem
.isNull() && destItem
.isDir()) {
400 const QMimeData
*mimeData
= QApplication::clipboard()->mimeData();
402 const QString text
= KIO::pasteActionText(mimeData
, &canPaste
, destItem
);
404 if (destItem
== m_fileInfo
) {
405 // if paste destination is a selected folder
406 action
= new QAction(QIcon::fromTheme(QStringLiteral("edit-paste")), text
, this);
407 connect(action
, &QAction::triggered
, m_mainWindow
, &DolphinMainWindow::pasteIntoFolder
);
409 action
= m_mainWindow
->actionCollection()->action(KStandardAction::name(KStandardAction::Paste
));
417 KFileItemListProperties
&DolphinContextMenu::selectedItemsProperties() const
419 if (!m_selectedItemsProperties
) {
420 m_selectedItemsProperties
= new KFileItemListProperties(m_selectedItems
);
422 return *m_selectedItemsProperties
;
425 KFileItem
DolphinContextMenu::baseFileItem()
427 if (!m_baseFileItem
) {
428 const DolphinView
*view
= m_mainWindow
->activeViewContainer()->view();
429 KFileItem baseItem
= view
->rootItem();
430 if (baseItem
.isNull() || baseItem
.url() != m_baseUrl
) {
431 m_baseFileItem
= new KFileItem(m_baseUrl
);
433 m_baseFileItem
= new KFileItem(baseItem
);
436 return *m_baseFileItem
;
439 void DolphinContextMenu::addOpenWithActions()
441 // insert 'Open With...' action or sub menu
442 m_fileItemActions
->insertOpenWithActionsTo(nullptr, this, QStringList
{qApp
->desktopFileName()});
445 void DolphinContextMenu::addAdditionalActions(const KFileItemListProperties
&props
)
449 QList
<QAction
*> additionalActions
;
450 if (props
.isLocal() && ContextMenuSettings::showOpenTerminal()) {
451 additionalActions
<< m_mainWindow
->actionCollection()->action(QStringLiteral("open_terminal_here"));
453 m_fileItemActions
->addActionsTo(this, KFileItemActions::MenuActionSource::All
, additionalActions
);
455 const DolphinView
*view
= m_mainWindow
->activeViewContainer()->view();
456 const QList
<QAction
*> versionControlActions
= view
->versionControlActions(m_selectedItems
);
457 if (!versionControlActions
.isEmpty()) {
458 addActions(versionControlActions
);