X-Git-Url: https://cloud.milkyroute.net/gitweb/dolphin.git/blobdiff_plain/c636d3e0edb3d6b84e492950c85ff8678ae5b1e4..refs/heads/master:/src/dolphincontextmenu.cpp diff --git a/src/dolphincontextmenu.cpp b/src/dolphincontextmenu.cpp index b25ff55a1..b6c70b48a 100644 --- a/src/dolphincontextmenu.cpp +++ b/src/dolphincontextmenu.cpp @@ -1,642 +1,517 @@ -/*************************************************************************** - * Copyright (C) 2006 by Peter Penz (peter.penz@gmx.at) and * - * Cvetoslav Ludmiloff * - * * - * This program is free software; you can redistribute it and/or modify * - * it under the terms of the GNU General Public License as published by * - * the Free Software Foundation; either version 2 of the License, or * - * (at your option) any later version. * - * * - * This program is distributed in the hope that it will be useful, * - * but WITHOUT ANY WARRANTY; without even the implied warranty of * - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * - * GNU General Public License for more details. * - * * - * You should have received a copy of the GNU General Public License * - * along with this program; if not, write to the * - * Free Software Foundation, Inc., * - * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * - ***************************************************************************/ +/* + * SPDX-FileCopyrightText: 2006 Peter Penz (peter.penz@gmx.at) and Cvetoslav Ludmiloff + * + * SPDX-License-Identifier: GPL-2.0-or-later + */ #include "dolphincontextmenu.h" +#include "dolphin_contextmenusettings.h" +#include "dolphin_generalsettings.h" #include "dolphinmainwindow.h" -#include "dolphinsettings.h" -#include "dolphinview.h" -#include "editbookmarkdialog.h" - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include +#include "dolphinnewfilemenu.h" +#include "dolphinplacesmodelsingleton.h" +#include "dolphinremoveaction.h" +#include "dolphinviewcontainer.h" +#include "global.h" +#include "trash/dolphintrash.h" +#include "views/dolphinview.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include #include #include -#include - -DolphinContextMenu::DolphinContextMenu(DolphinMainWindow* parent, - KFileItem* fileInfo, - const KUrl& baseUrl, - ViewType viewType) : - m_mainWindow(parent), - m_fileInfo(fileInfo), - m_baseUrl(baseUrl), - m_viewType(viewType), - m_context(NoContext) +#include +#include + +DolphinContextMenu::DolphinContextMenu(DolphinMainWindow *parent, + const KFileItem &fileInfo, + const KFileItemList &selectedItems, + const QUrl &baseUrl, + KFileItemActions *fileItemActions) + : QMenu(parent) + , m_mainWindow(parent) + , m_fileInfo(fileInfo) + , m_baseUrl(baseUrl) + , m_baseFileItem(nullptr) + , m_selectedItems(selectedItems) + , m_selectedItemsProperties(nullptr) + , m_context(NoContext) + , m_copyToMenu(parent) + , m_removeAction(nullptr) + , m_fileItemActions(fileItemActions) { - if (viewType == ItemsView) { - // The context menu either accesses the URLs of the selected items - // or the items itself. To increase the performance both lists are cached. - DolphinView* view = m_mainWindow->activeView(); - m_selectedUrls = view->selectedUrls(); - m_selectedItems = view->selectedItems(); - } - else if (fileInfo != 0) { - m_selectedUrls.append(fileInfo->url()); - m_selectedItems.append(fileInfo); - } + QApplication::instance()->installEventFilter(this); + + addAllActions(); } DolphinContextMenu::~DolphinContextMenu() { + delete m_baseFileItem; + m_baseFileItem = nullptr; + delete m_selectedItemsProperties; + m_selectedItemsProperties = nullptr; } -void DolphinContextMenu::open() +void DolphinContextMenu::addAllActions() { + static_cast(m_mainWindow->actionCollection()->action(QStringLiteral("hamburger_menu")))->addToMenu(this); + // get the context information - if (m_baseUrl.protocol() == "trash") { + const auto scheme = m_baseUrl.scheme(); + if (scheme == QLatin1String("trash")) { m_context |= TrashContext; + } else if (scheme.contains(QLatin1String("search"))) { + m_context |= SearchContext; + } else if (scheme.contains(QLatin1String("timeline"))) { + m_context |= TimelineContext; + } else if (scheme == QStringLiteral("recentlyused")) { + m_context |= RecentlyUsedContext; } - if (m_fileInfo != 0) { + if (!m_fileInfo.isNull() && !m_selectedItems.isEmpty()) { m_context |= ItemContext; // TODO: handle other use cases like devices + desktop files - } + } // open the corresponding popup for the context if (m_context & TrashContext) { if (m_context & ItemContext) { - openTrashItemContextMenu(); - } - else { - openTrashContextMenu(); + addTrashItemContextMenu(); + } else { + addTrashContextMenu(); } - } - else if (m_context & ItemContext) { - openItemContextMenu(); - } - else { - Q_ASSERT(m_context == NoContext); - openViewportContextMenu(); + } else if (m_context & ItemContext) { + addItemContextMenu(); + } else { + addViewportContextMenu(); } } -void DolphinContextMenu::cut() +bool DolphinContextMenu::eventFilter(QObject *object, QEvent *event) { - QMimeData* mimeData = new QMimeData(); - KUrl::List kdeUrls; - kdeUrls.append(m_fileInfo->url()); - KonqMimeData::populateMimeData(mimeData, kdeUrls, KUrl::List(), true); - QApplication::clipboard()->setMimeData(mimeData); -} + Q_UNUSED(object) -void DolphinContextMenu::copy() -{ - QMimeData* mimeData = new QMimeData(); - KUrl::List kdeUrls; - kdeUrls.append(m_fileInfo->url()); - KonqMimeData::populateMimeData(mimeData, kdeUrls, KUrl::List(), false); - QApplication::clipboard()->setMimeData(mimeData); -} + if (event->type() == QEvent::KeyPress || event->type() == QEvent::KeyRelease) { + QKeyEvent *keyEvent = static_cast(event); -void DolphinContextMenu::paste() -{ - QClipboard* clipboard = QApplication::clipboard(); - const QMimeData* mimeData = clipboard->mimeData(); - - const KUrl::List source = KUrl::List::fromMimeData(mimeData); - const KUrl& dest = m_fileInfo->url(); - if (KonqMimeData::decodeIsCutSelection(mimeData)) { - KonqOperations::copy(m_mainWindow, KonqOperations::MOVE, source, dest); - clipboard->clear(); - } - else { - KonqOperations::copy(m_mainWindow, KonqOperations::COPY, source, dest); + if (m_removeAction && keyEvent->key() == Qt::Key_Shift) { + if (event->type() == QEvent::KeyPress) { + m_removeAction->update(DolphinRemoveAction::ShiftState::Pressed); + } else { + m_removeAction->update(DolphinRemoveAction::ShiftState::Released); + } + } } -} -void DolphinContextMenu::rename() -{ - // TODO -} - -void DolphinContextMenu::moveToTrash() -{ - // TODO -} - -void DolphinContextMenu::deleteItem() -{ - // TODO -} - -void DolphinContextMenu::showProperties() -{ - new KPropertiesDialog(m_fileInfo->url()); + return false; } -void DolphinContextMenu::openTrashContextMenu() +void DolphinContextMenu::addTrashContextMenu() { Q_ASSERT(m_context & TrashContext); - KMenu* popup = new KMenu(m_mainWindow); - - QAction* emptyTrashAction = new QAction(KIcon("user-trash"), i18n("Emtpy Trash"), popup); - KConfig trashConfig("trashrc", KConfig::OnlyLocal); - emptyTrashAction->setEnabled(!trashConfig.group("Status").readEntry("Empty", true)); - popup->addAction(emptyTrashAction); - - QAction* propertiesAction = m_mainWindow->actionCollection()->action("properties"); - popup->addAction(propertiesAction); - - if (popup->exec(QCursor::pos()) == emptyTrashAction) { - const QString text(i18n("Do you really want to empty the Trash? All items will get deleted.")); - const bool del = KMessageBox::warningContinueCancel(m_mainWindow, - text, - QString(), - KGuiItem(i18n("Empty Trash"), KIcon("user-trash")) - ) == KMessageBox::Continue; - if (del) { - KonqOperations::emptyTrash(m_mainWindow); - } + QAction *emptyTrashAction = addAction(QIcon::fromTheme(QStringLiteral("edit-delete")), i18nc("@action:inmenu", "Empty Trash"), this, [this]() { + Trash::empty(m_mainWindow); + }); + emptyTrashAction->setEnabled(!Trash::isEmpty()); + + // Insert 'Sort By' and 'View Mode' + if (ContextMenuSettings::showSortBy() || ContextMenuSettings::showViewMode()) { + addSeparator(); + } + if (ContextMenuSettings::showSortBy()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("sort"))); + } + if (ContextMenuSettings::showViewMode()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("view_mode"))); } - popup->deleteLater(); + addSeparator(); + QAction *propertiesAction = m_mainWindow->actionCollection()->action(QStringLiteral("properties")); + addAction(propertiesAction); } -void DolphinContextMenu::openTrashItemContextMenu() +void DolphinContextMenu::addTrashItemContextMenu() { Q_ASSERT(m_context & TrashContext); Q_ASSERT(m_context & ItemContext); - KMenu* popup = new KMenu(m_mainWindow); + addAction(QIcon::fromTheme(QStringLiteral("edit-reset")), + i18ncp("@action:inmenu Restore the selected files that are in the trash to the place they lived at the moment they were trashed. Minimize the " + "length of this string if possible.", + "Restore to Former Location", + "Restore to Former Locations", + m_selectedItems.count()), + this, + [this]() { + QList selectedUrls; + selectedUrls.reserve(m_selectedItems.count()); + for (const KFileItem &item : std::as_const(m_selectedItems)) { + selectedUrls.append(item.url()); + } - QAction* restoreAction = new QAction(i18n("Restore"), m_mainWindow); - popup->addAction(restoreAction); + KIO::RestoreJob *job = KIO::restoreFromTrash(selectedUrls); + KJobWidgets::setWindow(job, m_mainWindow); + job->uiDelegate()->setAutoErrorHandlingEnabled(true); + }); - QAction* deleteAction = m_mainWindow->actionCollection()->action("delete"); - popup->addAction(deleteAction); + addSeparator(); - QAction* propertiesAction = m_mainWindow->actionCollection()->action("properties"); - popup->addAction(propertiesAction); + addAction(m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::Cut))); + addAction(m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::Copy))); - if (popup->exec(QCursor::pos()) == restoreAction) { - KonqOperations::restoreTrashedItems(m_selectedUrls, m_mainWindow); - } + addSeparator(); - popup->deleteLater(); -} + QAction *deleteAction = m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile)); + addAction(deleteAction); -void DolphinContextMenu::openItemContextMenu() -{ - Q_ASSERT(m_fileInfo != 0); + addSeparator(); - KMenu* popup = new KMenu(m_mainWindow); - insertDefaultItemActions(popup); + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("properties"))); +} - popup->addSeparator(); +void DolphinContextMenu::addDirectoryItemContextMenu() +{ + // insert 'Open in new window' and 'Open in new tab' entries + const KFileItemListProperties &selectedItemsProps = selectedItemsProperties(); + if (ContextMenuSettings::showOpenInNewTab()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("open_in_new_tab"))); + } + if (ContextMenuSettings::showOpenInNewWindow()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("open_in_new_window"))); + } - // insert 'Bookmark this folder' entry if exactly one item is selected - QAction* bookmarkAction = 0; - if (m_fileInfo->isDir() && (m_selectedUrls.count() == 1)) { - bookmarkAction = popup->addAction(KIcon("bookmark-folder"), i18n("Bookmark folder")); + if (ContextMenuSettings::showOpenInSplitView()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("open_in_split_view"))); } - // Insert 'Open With...' sub menu - QVector openWithVector; - const QList openWithActions = insertOpenWithItems(popup, openWithVector); + // Insert 'Open With' entries + addOpenWithActions(); - // Insert 'Actions' sub menu - QVector actionsVector; - const QList serviceActions = insertActionItems(popup, actionsVector); - popup->addSeparator(); + // set up 'Create New' menu + QAction *newDirAction = m_mainWindow->actionCollection()->action(QStringLiteral("create_dir")); + QAction *newFileAction = m_mainWindow->actionCollection()->action(QStringLiteral("create_file")); + DolphinNewFileMenu *newFileMenu = new DolphinNewFileMenu(newDirAction, newFileAction, this); + newFileMenu->checkUpToDate(); + newFileMenu->setWorkingDirectory(m_fileInfo.url()); + newFileMenu->setEnabled(selectedItemsProps.supportsWriting()); + connect(newFileMenu, &DolphinNewFileMenu::fileCreated, newFileMenu, &DolphinNewFileMenu::deleteLater); + connect(newFileMenu, &DolphinNewFileMenu::directoryCreated, newFileMenu, &DolphinNewFileMenu::deleteLater); - // insert 'Properties...' entry - QAction* propertiesAction = 0; - if (m_viewType == SidebarView) { - propertiesAction = new QAction(i18n("Properties..."), this); - connect(this, SIGNAL(triggered()), this, SLOT(showProperties())); - } - else { - propertiesAction = m_mainWindow->actionCollection()->action("properties"); - } - popup->addAction(propertiesAction); - - QAction* activatedAction = popup->exec(QCursor::pos()); - - if ((bookmarkAction!= 0) && (activatedAction == bookmarkAction)) { - const KUrl selectedUrl(m_fileInfo->url()); - KBookmark bookmark = EditBookmarkDialog::getBookmark(i18n("Add folder as bookmark"), - selectedUrl.fileName(), - selectedUrl, - "bookmark"); - if (!bookmark.isNull()) { - KBookmarkManager* manager = DolphinSettings::instance().bookmarkManager(); - KBookmarkGroup root = manager->root(); - root.addBookmark(manager, bookmark); - manager->emitChanged(root); - } - } - else if (serviceActions.contains(activatedAction)) { - // one of the 'Actions' items has been selected - int id = serviceActions.indexOf(activatedAction); - KDEDesktopMimeType::executeService(m_selectedUrls, actionsVector[id]); - } - else if (openWithActions.contains(activatedAction)) { - // one of the 'Open With' items has been selected - if (openWithActions.last() == activatedAction) { - // the item 'Other...' has been selected - KRun::displayOpenWithDialog(m_selectedUrls, m_mainWindow); - } - else { - int id = openWithActions.indexOf(activatedAction); - KService::Ptr servicePtr = openWithVector[id]; - KRun::run(*servicePtr, m_selectedUrls, m_mainWindow); - } - } + QMenu *menu = newFileMenu->menu(); + menu->setTitle(i18nc("@title:menu Create new folder, file, link, etc.", "Create New")); + menu->setIcon(QIcon::fromTheme(QStringLiteral("list-add"))); + addMenu(menu); - openWithVector.clear(); - actionsVector.clear(); - popup->deleteLater(); + addSeparator(); } -void DolphinContextMenu::openViewportContextMenu() +void DolphinContextMenu::addOpenParentFolderActions() { - Q_ASSERT(m_fileInfo == 0); - KMenu* popup = new KMenu(m_mainWindow); + addAction(QIcon::fromTheme(QStringLiteral("document-open-folder")), i18nc("@action:inmenu", "Open Path"), [this]() { + const QUrl url = m_fileInfo.targetUrl(); + const QUrl parentUrl = KIO::upUrl(url); + m_mainWindow->changeUrl(parentUrl); + m_mainWindow->activeViewContainer()->view()->markUrlsAsSelected({url}); + m_mainWindow->activeViewContainer()->view()->markUrlAsCurrent(url); + }); + + addAction(QIcon::fromTheme(QStringLiteral("tab-new")), i18nc("@action:inmenu", "Open Path in New Tab"), [this]() { + const QUrl url = m_fileInfo.targetUrl(); + const QUrl parentUrl = KIO::upUrl(url); + DolphinTabPage *tabPage = m_mainWindow->openNewTab(parentUrl); + tabPage->activeViewContainer()->view()->markUrlsAsSelected({url}); + tabPage->activeViewContainer()->view()->markUrlAsCurrent(url); + }); + + addAction(QIcon::fromTheme(QStringLiteral("window-new")), i18nc("@action:inmenu", "Open Path in New Window"), [this]() { + Dolphin::openNewWindow({m_fileInfo.targetUrl()}, m_mainWindow, Dolphin::OpenNewWindowFlag::Select); + }); +} - // setup 'Create New' menu - KNewMenu* newMenu = m_mainWindow->newMenu(); - newMenu->slotCheckUpToDate(); - newMenu->setPopupFiles(m_baseUrl); - popup->addMenu(newMenu->menu()); - popup->addSeparator(); +void DolphinContextMenu::addItemContextMenu() +{ + Q_ASSERT(!m_fileInfo.isNull()); - QAction* pasteAction = m_mainWindow->actionCollection()->action(KStandardAction::stdName(KStandardAction::Paste)); - popup->addAction(pasteAction); + const KFileItemListProperties &selectedItemsProps = selectedItemsProperties(); - // setup 'View Mode' menu - KMenu* viewModeMenu = new KMenu(i18n("View Mode")); + m_fileItemActions->setItemListProperties(selectedItemsProps); - QAction* iconsMode = m_mainWindow->actionCollection()->action("icons"); - viewModeMenu->addAction(iconsMode); + if (m_selectedItems.count() == 1) { + // single files + if (m_fileInfo.isDir()) { + addDirectoryItemContextMenu(); + } else if (m_context & TimelineContext || m_context & SearchContext || m_context & RecentlyUsedContext) { + addOpenWithActions(); - QAction* detailsMode = m_mainWindow->actionCollection()->action("details"); - viewModeMenu->addAction(detailsMode); + addOpenParentFolderActions(); - QAction* previewsMode = m_mainWindow->actionCollection()->action("previews"); - viewModeMenu->addAction(previewsMode); + addSeparator(); + } else { + // Insert 'Open With" entries + addOpenWithActions(); + } + if (m_fileInfo.isLink()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("show_target"))); + addSeparator(); + } + } else { + // multiple files + bool selectionHasOnlyDirs = true; + for (const auto &item : std::as_const(m_selectedItems)) { + const QUrl &url = DolphinView::openItemAsFolderUrl(item); + if (url.isEmpty()) { + selectionHasOnlyDirs = false; + break; + } + } - popup->addMenu(viewModeMenu); - popup->addSeparator(); + if (selectionHasOnlyDirs && ContextMenuSettings::showOpenInNewTab()) { + // insert 'Open in new tab' entry + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("open_in_new_tabs"))); + } + // Insert 'Open With" entries + addOpenWithActions(); + } - QAction* bookmarkAction = popup->addAction(KIcon("bookmark-folder"), i18n("Bookmark this folder")); - popup->addSeparator(); + insertDefaultItemActions(selectedItemsProps); - QAction* propertiesAction = popup->addAction(i18n("Properties...")); + addAdditionalActions(selectedItemsProps); - QAction* activatedAction = popup->exec(QCursor::pos()); - if (activatedAction == propertiesAction) { - new KPropertiesDialog(m_mainWindow->activeView()->url()); + // insert 'Copy To' and 'Move To' sub menus + if (ContextMenuSettings::showCopyMoveMenu()) { + m_copyToMenu.setUrls(m_selectedItems.urlList()); + m_copyToMenu.setReadOnly(!selectedItemsProps.supportsWriting()); + m_copyToMenu.setAutoErrorHandlingEnabled(true); + m_copyToMenu.addActionsTo(this); } - else if (activatedAction == bookmarkAction) { - const KUrl& url = m_mainWindow->activeView()->url(); - KBookmark bookmark = EditBookmarkDialog::getBookmark(i18n("Add folder as bookmark"), - url.fileName(), - url, - "bookmark"); - if (!bookmark.isNull()) { - KBookmarkManager* manager = DolphinSettings::instance().bookmarkManager(); - KBookmarkGroup root = manager->root(); - root.addBookmark(manager, bookmark); - manager->emitChanged(root); + + if (m_mainWindow->isSplitViewEnabledInCurrentTab()) { + if (ContextMenuSettings::showCopyToOtherSplitView()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("copy_to_inactive_split_view"))); + } + + if (ContextMenuSettings::showMoveToOtherSplitView()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("move_to_inactive_split_view"))); } } - popup->deleteLater(); + // insert 'Properties...' entry + addSeparator(); + QAction *propertiesAction = m_mainWindow->actionCollection()->action(QStringLiteral("properties")); + addAction(propertiesAction); } -void DolphinContextMenu::insertDefaultItemActions(KMenu* popup) +void DolphinContextMenu::addViewportContextMenu() { - Q_ASSERT(popup != 0); - const KActionCollection* collection = m_mainWindow->actionCollection(); - const bool insertSidebarActions = (m_viewType == SidebarView); - - // insert 'Cut', 'Copy' and 'Paste' - QAction* cutAction = 0; - QAction* copyAction = 0; - QAction* pasteAction = 0; - if (insertSidebarActions) { - cutAction = new QAction(KIcon("edit-cut"), i18n("Cut"), this); - connect(cutAction, SIGNAL(triggered()), this, SLOT(cut())); - - copyAction = new QAction(KIcon("edit-copy"), i18n("Copy"), this); - connect(copyAction, SIGNAL(triggered()), this, SLOT(copy())); - - const QAction* menuPasteAction = collection->action(KStandardAction::stdName(KStandardAction::Paste)); - pasteAction = new QAction(KIcon("edit-paste"), menuPasteAction->text(), this); - pasteAction->setEnabled(menuPasteAction->isEnabled()); - connect(pasteAction, SIGNAL(triggered()), this, SLOT(paste())); - } - else { - cutAction = collection->action(KStandardAction::stdName(KStandardAction::Cut)); - copyAction = collection->action(KStandardAction::stdName(KStandardAction::Copy)); - pasteAction = collection->action(KStandardAction::stdName(KStandardAction::Paste)); + const KFileItemListProperties baseUrlProperties(KFileItemList() << baseFileItem()); + m_fileItemActions->setItemListProperties(baseUrlProperties); + + // Set up and insert 'Create New' menu + KNewFileMenu *newFileMenu = m_mainWindow->newFileMenu(); + newFileMenu->checkUpToDate(); + newFileMenu->setWorkingDirectory(m_baseUrl); + addMenu(newFileMenu->menu()); + + // Show "open with" menu items even if the dir is empty, because there are legitimate + // use cases for this, such as opening an empty dir in Kate or VSCode or something + addOpenWithActions(); + + QAction *pasteAction = createPasteAction(); + if (pasteAction) { + addAction(pasteAction); } - popup->addAction(cutAction); - popup->addAction(copyAction); - popup->addAction(pasteAction); - popup->addSeparator(); + // Insert 'Add to Places' entry if it's not already in the places panel + if (ContextMenuSettings::showAddToPlaces() && !placeExists(m_mainWindow->activeViewContainer()->url())) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("add_to_places"))); + } + addSeparator(); - // insert 'Rename' - QAction* renameAction = 0; - if (insertSidebarActions) { - renameAction = new QAction(i18n("Rename"), this); - connect(renameAction, SIGNAL(triggered()), this, SLOT(paste())); + // Insert 'Sort By', 'Group By' and 'View Mode' + if (ContextMenuSettings::showSortBy()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("sort"))); } - else { - renameAction = collection->action("rename"); + if (ContextMenuSettings::showGroupBy()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("group"))); } - popup->addAction(renameAction); - - // insert 'Move to Trash' and (optionally) 'Delete' - const KSharedConfig::Ptr globalConfig = KSharedConfig::openConfig("kdeglobals", KConfig::NoGlobals); - const KConfigGroup kdeConfig(globalConfig, "KDE"); - bool showDeleteCommand = kdeConfig.readEntry("ShowDeleteCommand", false); - const KUrl& url = insertSidebarActions ? m_fileInfo->url(): - m_mainWindow->activeView()->url(); - if (url.isLocalFile()) { - QAction* moveToTrashAction = 0; - if (insertSidebarActions) { - moveToTrashAction = new QAction(KIcon("edit-trash"), i18n("Move To Trash"), this); - connect(moveToTrashAction, SIGNAL(triggered()), this, SLOT(moveToTrash())); - } - else { - moveToTrashAction = collection->action("move_to_trash"); - } - popup->addAction(moveToTrashAction); + if (ContextMenuSettings::showViewMode()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("view_mode"))); } - else { - showDeleteCommand = true; + if (ContextMenuSettings::showSortBy() || ContextMenuSettings::showViewMode()) { + addSeparator(); } - if (showDeleteCommand) { - QAction* deleteAction = 0; - if (insertSidebarActions) { - deleteAction = new QAction(KIcon("edit-delete"), i18n("Delete"), this); - connect(deleteAction, SIGNAL(triggered()), this, SLOT(deleteItem())); - } - else { - deleteAction = collection->action("delete"); - } - popup->addAction(deleteAction); - } + addAdditionalActions(baseUrlProperties); + + addSeparator(); + + QAction *propertiesAction = m_mainWindow->actionCollection()->action(QStringLiteral("properties")); + addAction(propertiesAction); } -QList DolphinContextMenu::insertOpenWithItems(KMenu* popup, - QVector& openWithVector) +void DolphinContextMenu::insertDefaultItemActions(const KFileItemListProperties &properties) { - // Parts of the following code have been taken - // from the class KonqOperations located in - // libqonq/konq_operations.h of Konqueror. - // (Copyright (C) 2000 David Faure ) - - // Prepare 'Open With' sub menu. Usually a sub menu is created, where all applications - // are listed which are registered to open the item. As last entry "Other..." will be - // attached which allows to select a custom application. If no applications are registered - // no sub menu is created at all, only "Open With..." will be offered. - bool insertOpenWithItems = true; - const QString contextMimeType(m_fileInfo->mimetype()); - - QListIterator mimeIt(m_selectedItems); - while (insertOpenWithItems && mimeIt.hasNext()) { - KFileItem* item = mimeIt.next(); - insertOpenWithItems = (contextMimeType == item->mimetype()); + const KActionCollection *collection = m_mainWindow->actionCollection(); + + // Insert 'Cut', 'Copy', 'Copy Location' and 'Paste' + addAction(collection->action(KStandardAction::name(KStandardAction::Cut))); + addAction(collection->action(KStandardAction::name(KStandardAction::Copy))); + if (ContextMenuSettings::showCopyLocation()) { + QAction *copyPathAction = collection->action(QStringLiteral("copy_location")); + copyPathAction->setEnabled(m_selectedItems.size() == 1); + addAction(copyPathAction); + } + QAction *pasteAction = createPasteAction(); + if (pasteAction) { + addAction(pasteAction); } - QList openWithActions; - if (insertOpenWithItems) { - // fill the 'Open with' sub menu with application types - const KMimeType::Ptr mimePtr = KMimeType::findByUrl(m_fileInfo->url()); - KService::List offers = KMimeTypeTrader::self()->query(mimePtr->name(), - "Application", - "Type == 'Application'"); - if (offers.count() > 0) { - KService::List::Iterator it; - KMenu* openWithMenu = new KMenu(i18n("Open With")); - for(it = offers.begin(); it != offers.end(); ++it) { - // The offer list from the KTrader returns duplicate - // application entries. Although this seems to be a configuration - // problem outside the scope of Dolphin, duplicated entries just - // will be skipped here. - const QString appName((*it)->name()); - if (!containsEntry(openWithMenu, appName)) { - const KIcon icon((*it)->icon()); - QAction* action = openWithMenu->addAction(icon, appName); - openWithVector.append(*it); - openWithActions << action; - } - } + // Insert 'Duplicate Here' + if (ContextMenuSettings::showDuplicateHere()) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("duplicate"))); + } - openWithMenu->addSeparator(); - QAction* action = openWithMenu->addAction(i18n("&Other...")); + // Insert 'Rename' + addAction(collection->action(KStandardAction::name(KStandardAction::RenameFile))); - openWithActions << action; - popup->addMenu(openWithMenu); - } - else { - // No applications are registered, hence just offer - // a "Open With..." item instead of a sub menu containing - // only one entry. - QAction* action = popup->addAction(i18n("Open With...")); - openWithActions << action; - } + // Insert 'Add to Places' entry if appropriate + if (ContextMenuSettings::showAddToPlaces() && m_selectedItems.count() == 1 && m_fileInfo.isDir() && !placeExists(m_fileInfo.url())) { + addAction(m_mainWindow->actionCollection()->action(QStringLiteral("add_to_places"))); } - else { - // At least one of the selected items has a different MIME type. In this case - // just show a disabled "Open With..." entry. - QAction* action = popup->addAction(i18n("Open With...")); - action->setEnabled(false); + + addSeparator(); + + // Insert 'Move to Trash' and/or 'Delete' + const bool showDeleteAction = (KSharedConfig::openConfig()->group(QStringLiteral("KDE")).readEntry("ShowDeleteCommand", false) || !properties.isLocal()); + const bool showMoveToTrashAction = (properties.isLocal() && properties.supportsMoving()); + + if (showDeleteAction && showMoveToTrashAction) { + delete m_removeAction; + m_removeAction = nullptr; + addAction(m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::MoveToTrash))); + addAction(m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile))); + } else if (showDeleteAction && !showMoveToTrashAction) { + addAction(m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::DeleteFile))); + } else { + if (!m_removeAction) { + m_removeAction = new DolphinRemoveAction(this, m_mainWindow->actionCollection()); + } + addAction(m_removeAction); + m_removeAction->update(); } +} + +bool DolphinContextMenu::placeExists(const QUrl &url) const +{ + const KFilePlacesModel *placesModel = DolphinPlacesModelSingleton::instance().placesModel(); - return openWithActions; + QModelIndex url_index = placesModel->closestItem(url); + return url_index.isValid() && placesModel->url(url_index).matches(url, QUrl::StripTrailingSlash); } -QList DolphinContextMenu::insertActionItems(KMenu* popup, - QVector& actionsVector) +QAction *DolphinContextMenu::createPasteAction() { - // Parts of the following code have been taken - // from the class KonqOperations located in - // libqonq/konq_operations.h of Konqueror. - // (Copyright (C) 2000 David Faure ) - - KMenu* actionsMenu = new KMenu(i18n("Actions")); - - QList serviceActions; - - QStringList dirs = KGlobal::dirs()->findDirs("data", "dolphin/servicemenus/"); - - KMenu* menu = 0; - for (QStringList::ConstIterator dirIt = dirs.begin(); dirIt != dirs.end(); ++dirIt) { - QDir dir(*dirIt); - QStringList filters; - filters << "*.desktop"; - dir.setNameFilters(filters); - QStringList entries = dir.entryList(QDir::Files); - - for (QStringList::ConstIterator entryIt = entries.begin(); entryIt != entries.end(); ++entryIt) { - KConfigGroup cfg(KSharedConfig::openConfig( *dirIt + *entryIt, KConfig::OnlyLocal), "Desktop Entry" ); - if ((cfg.hasKey("Actions") || cfg.hasKey("X-KDE-GetActionMenu")) && cfg.hasKey("ServiceTypes")) { - //const QStringList types = cfg.readListEntry("ServiceTypes"); - QStringList types; - types = cfg.readEntry("ServiceTypes", types); - for (QStringList::ConstIterator it = types.begin(); it != types.end(); ++it) { - // check whether the mime type is equal or whether the - // mimegroup (e. g. image/*) is supported - - bool insert = false; - if ((*it) == "all/allfiles") { - // The service type is valid for all files, but not for directories. - // Check whether the selected items only consist of files... - QListIterator mimeIt(m_selectedItems); - insert = true; - while (insert && mimeIt.hasNext()) { - KFileItem* item = mimeIt.next(); - insert = !item->isDir(); - } - } - - if (!insert) { - // Check whether the MIME types of all selected files match - // to the mimetype of the service action. As soon as one MIME - // type does not match, no service menu is shown at all. - QListIterator mimeIt(m_selectedItems); - insert = true; - while (insert && mimeIt.hasNext()) { - KFileItem* item = mimeIt.next(); - const QString mimeType(item->mimetype()); - const QString mimeGroup(mimeType.left(mimeType.indexOf('/'))); - - insert = (*it == mimeType) || - ((*it).right(1) == "*") && - ((*it).left((*it).indexOf('/')) == mimeGroup); - } - } - - if (insert) { - menu = actionsMenu; - - const QString submenuName = cfg.readEntry( "X-KDE-Submenu" ); - if (!submenuName.isEmpty()) { - menu = new KMenu(submenuName); - actionsMenu->addMenu(menu); - } - - Q3ValueList userServices = - KDEDesktopMimeType::userDefinedServices(*dirIt + *entryIt, true); - - Q3ValueList::Iterator serviceIt; - for (serviceIt = userServices.begin(); serviceIt != userServices.end(); ++serviceIt) { - KDEDesktopMimeType::Service service = (*serviceIt); - if (!service.m_strIcon.isEmpty()) { - QAction* action = menu->addAction(SmallIcon(service.m_strIcon), - service.m_strName); - serviceActions << action; - } - else { - QAction *action = menu->addAction(service.m_strName); - serviceActions << action; - } - actionsVector.append(service); - } - } - } + QAction *action = nullptr; + KFileItem destItem; + if (!m_fileInfo.isNull() && m_selectedItems.count() <= 1) { + destItem = m_fileInfo; + } else { + destItem = baseFileItem(); + } + + if (!destItem.isNull() && destItem.isDir()) { + const QMimeData *mimeData = QApplication::clipboard()->mimeData(); + bool canPaste; + const QString text = KIO::pasteActionText(mimeData, &canPaste, destItem); + if (canPaste) { + if (destItem == m_fileInfo) { + // if paste destination is a selected folder + action = new QAction(QIcon::fromTheme(QStringLiteral("edit-paste")), text, this); + connect(action, &QAction::triggered, m_mainWindow, &DolphinMainWindow::pasteIntoFolder); + } else { + action = m_mainWindow->actionCollection()->action(KStandardAction::name(KStandardAction::Paste)); } } } - const int itemsCount = actionsMenu->actions().count(); - if (itemsCount == 0) { - // no actions are available at all, hence show the "Actions" - // submenu disabled - actionsMenu->setEnabled(false); + return action; +} + +KFileItemListProperties &DolphinContextMenu::selectedItemsProperties() const +{ + if (!m_selectedItemsProperties) { + m_selectedItemsProperties = new KFileItemListProperties(m_selectedItems); } + return *m_selectedItemsProperties; +} - if (itemsCount == 1) { - // Exactly one item is available. Instead of showing a sub menu with - // only one item, show the item directly in the root menu. - if (menu == actionsMenu) { - // The item is an action, hence show the action in the root menu. - const QList actions = actionsMenu->actions(); - Q_ASSERT(actions.count() == 1); - - const QString text = actions[0]->text(); - const QIcon icon = actions[0]->icon(); - if (icon.isNull()) { - QAction* action = popup->addAction(text); - serviceActions.clear(); - serviceActions << action; - } - else { - QAction* action = popup->addAction(icon, text); - serviceActions.clear(); - serviceActions << action; - } - } - else { - // The item is a sub menu, hence show the sub menu in the root menu. - popup->addMenu(menu); +KFileItem DolphinContextMenu::baseFileItem() +{ + if (!m_baseFileItem) { + const DolphinView *view = m_mainWindow->activeViewContainer()->view(); + KFileItem baseItem = view->rootItem(); + if (baseItem.isNull() || baseItem.url() != m_baseUrl) { + m_baseFileItem = new KFileItem(m_baseUrl); + } else { + m_baseFileItem = new KFileItem(baseItem); } - actionsMenu->deleteLater(); - actionsMenu = 0; - } - else { - popup->addMenu(actionsMenu); } - - return serviceActions; + return *m_baseFileItem; } -bool DolphinContextMenu::containsEntry(const KMenu* menu, - const QString& entryName) const +void DolphinContextMenu::addOpenWithActions() { - Q_ASSERT(menu != 0); - - const QList list = menu->actions(); - const uint count = list.count(); - for (uint i = 0; i < count; ++i) { - const QAction* action = list.at(i); - if (action->text() == entryName) { - return true; + // insert 'Open With...' action or sub menu + m_fileItemActions->insertOpenWithActionsTo(nullptr, this, QStringList{qApp->desktopFileName()}); + + // For a single file, hint in "Open with" menu that middle-clicking would open it in the secondary app. + // (Unless middle-clicking would open it as a folder in a new tab (e.g. archives).) + const QUrl &url = DolphinView::openItemAsFolderUrl(m_fileInfo, GeneralSettings::browseThroughArchives()); + if (m_selectedItems.count() == 1 && url.isEmpty()) { + if (QAction *openWithSubMenu = findChild(QStringLiteral("openWith_submenu"))) { + Q_ASSERT(openWithSubMenu->menu()); + Q_ASSERT(!openWithSubMenu->menu()->isEmpty()); + + auto *secondaryApp = openWithSubMenu->menu()->actions().first(); + // Add it like a keyboard shortcut, Qt uses \t as a separator. + if (!secondaryApp->text().contains(QLatin1Char('\t'))) { + secondaryApp->setText(secondaryApp->text() + QLatin1Char('\t') + + i18nc("@action:inmenu Shortcut, middle click to trigger menu item, keep short", "Middle Click")); + } } } +} + +void DolphinContextMenu::addAdditionalActions(const KFileItemListProperties &props) +{ + addSeparator(); + + QList additionalActions; + if (props.isLocal() && ContextMenuSettings::showOpenTerminal()) { + additionalActions << m_mainWindow->actionCollection()->action(QStringLiteral("open_terminal_here")); + } + m_fileItemActions->addActionsTo(this, KFileItemActions::MenuActionSource::All, additionalActions); - return false; + const DolphinView *view = m_mainWindow->activeViewContainer()->view(); + const QList versionControlActions = view->versionControlActions(m_selectedItems); + if (!versionControlActions.isEmpty()) { + addSeparator(); + addActions(versionControlActions); + addSeparator(); + } } -#include "dolphincontextmenu.moc" +#include "moc_dolphincontextmenu.cpp"