okular/shell/shell.cpp
Sune Vuorela 720e08220d Allow dbus to be disabled
In certain usecases, especially for non-linux desktop builds, the
advantage of of dbus is often limited. Provide a way to disable
building with dbus.

Note that this also disables the related functionalities of having a
second okular launch open the document in the first okular.

This also disables the unit tests that tests those features.
2023-07-18 07:53:42 +00:00

1104 lines
37 KiB
C++

/*
SPDX-FileCopyrightText: 2002 Wilco Greven <greven@kde.org>
SPDX-FileCopyrightText: 2002 Chris Cheney <ccheney@cheney.cx>
SPDX-FileCopyrightText: 2003 Benjamin Meyer <benjamin@csh.rit.edu>
SPDX-FileCopyrightText: 2003-2004 Christophe Devriese <Christophe.Devriese@student.kuleuven.ac.be>
SPDX-FileCopyrightText: 2003 Laurent Montel <montel@kde.org>
SPDX-FileCopyrightText: 2003-2004 Albert Astals Cid <aacid@kde.org>
SPDX-FileCopyrightText: 2003 Luboš Luňák <l.lunak@kde.org>
SPDX-FileCopyrightText: 2003 Malcolm Hunter <malcolm.hunter@gmx.co.uk>
SPDX-FileCopyrightText: 2004 Dominique Devriese <devriese@kde.org>
SPDX-FileCopyrightText: 2004 Dirk Mueller <mueller@kde.org>
Work sponsored by the LiMux project of the city of Munich:
SPDX-FileCopyrightText: 2017 Klarälvdalens Datakonsult AB a KDAB Group company <info@kdab.com>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "shell.h"
// qt/kde includes
#include <KActionCollection>
#include <KConfigGroup>
#include <KIO/Global>
#include <KLocalizedString>
#include <KMessageBox>
#include <KPluginFactory>
#include <KPluginLoader>
#include <KRecentFilesAction>
#include <KSharedConfig>
#include <KStandardAction>
#include <KStartupInfo>
#include <KToggleFullScreenAction>
#include <KToolBar>
#include <KUrlMimeData>
#include <KWindowSystem>
#include <KXMLGUIFactory>
#include <QApplication>
#if HAVE_DBUS
#include <QDBusConnection>
#endif // HAVE_DBUS
#include <QDockWidget>
#include <QDragMoveEvent>
#include <QFileDialog>
#include <QJsonArray>
#include <QMenuBar>
#include <QMimeData>
#include <QObject>
#include <QScreen>
#include <QTabBar>
#include <QTabWidget>
#include <QTimer>
#ifdef WITH_KACTIVITIES
#include <KActivities/ResourceInstance>
#endif
#include <kio_version.h>
#include <kxmlgui_version.h>
// local includes
#include "../interfaces/viewerinterface.h"
#include "kdocumentviewer.h"
#include "shellutils.h"
static const char *shouldShowMenuBarComingFromFullScreen = "shouldShowMenuBarComingFromFullScreen";
static const char *shouldShowToolBarComingFromFullScreen = "shouldShowToolBarComingFromFullScreen";
static const char *const SESSION_URL_KEY = "Urls";
static const char *const SESSION_TAB_KEY = "ActiveTab";
static constexpr char SIDEBAR_LOCKED_KEY[] = "LockSidebar";
static constexpr char SIDEBAR_VISIBLE_KEY[] = "ShowSidebar";
/**
* Groups sidebar containers in a QDockWidget.
*
* This control groups all the sidebar containers provided by each tab (the Part object),
* allowing the user to dock it to the left and right sides of the window,
* or detach it from the window altogether.
*/
class Sidebar : public QDockWidget
{
Q_OBJECT
public:
explicit Sidebar(QWidget *parent = nullptr)
: QDockWidget(parent)
{
setAllowedAreas(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea);
setFeatures(defaultFeatures());
m_stackedWidget = new QStackedWidget;
setWidget(m_stackedWidget);
}
bool isLocked() const
{
return features().testFlag(NoDockWidgetFeatures);
}
void setLocked(bool locked)
{
setFeatures(locked ? NoDockWidgetFeatures : defaultFeatures());
// show titlebar only if not locked
if (locked) {
if (!m_dumbTitleWidget) {
m_dumbTitleWidget = new QWidget;
}
setTitleBarWidget(m_dumbTitleWidget);
} else {
setTitleBarWidget(nullptr);
}
}
int indexOf(QWidget *widget) const
{
return m_stackedWidget->indexOf(widget);
}
void addWidget(QWidget *widget)
{
m_stackedWidget->addWidget(widget);
}
void removeWidget(QWidget *widget)
{
m_stackedWidget->removeWidget(widget);
}
void setCurrentWidget(QWidget *widget)
{
m_stackedWidget->setCurrentWidget(widget);
}
private:
static DockWidgetFeatures defaultFeatures()
{
DockWidgetFeatures dockFeatures = DockWidgetClosable | DockWidgetMovable;
if (!KWindowSystem::isPlatformWayland()) { // TODO : Remove this check when QTBUG-87332 is fixed
dockFeatures |= DockWidgetFloatable;
}
return dockFeatures;
}
QStackedWidget *m_stackedWidget = nullptr;
QWidget *m_dumbTitleWidget = nullptr;
};
Shell::Shell(const QString &serializedOptions)
: KParts::MainWindow()
, m_menuBarWasShown(true)
, m_toolBarWasShown(true)
#ifndef Q_OS_WIN
, m_activityResource(nullptr)
#endif
, m_isValid(true)
{
setObjectName(QStringLiteral("okular::Shell#"));
setContextMenuPolicy(Qt::NoContextMenu);
// otherwise .rc file won't be found by unit test
setComponentName(QStringLiteral("okular"), QString());
// set the shell's ui resource file
setXMLFile(QStringLiteral("shell.rc"));
m_fileformatsscanned = false;
m_showMenuBarAction = nullptr;
// this routine will find and load our Part. it finds the Part by
// name which is a bad idea usually.. but it's alright in this
// case since our Part is made for this Shell
KPluginLoader loader(QStringLiteral("okularpart"));
m_partFactory = loader.factory();
if (!m_partFactory) {
// if we couldn't find our Part, we exit since the Shell by
// itself can't do anything useful
m_isValid = false;
KMessageBox::error(this, i18n("Unable to find the Okular component: %1", loader.errorString()));
return;
}
// now that the Part plugin is loaded, create the part
KParts::ReadWritePart *const firstPart = m_partFactory->create<KParts::ReadWritePart>(this);
if (firstPart) {
// Setup the central widget
m_centralStackedWidget = new QStackedWidget();
setCentralWidget(m_centralStackedWidget);
// Setup the welcome screen
m_welcomeScreen = new WelcomeScreen(this);
connect(m_welcomeScreen, &WelcomeScreen::openClicked, this, &Shell::fileOpen);
connect(m_welcomeScreen, &WelcomeScreen::closeClicked, this, &Shell::hideWelcomeScreen);
connect(m_welcomeScreen, &WelcomeScreen::recentItemClicked, this, [this](const QUrl &url) { openUrl(url); });
connect(m_welcomeScreen, &WelcomeScreen::forgetRecentItem, this, &Shell::forgetRecentItem);
m_centralStackedWidget->addWidget(m_welcomeScreen);
m_welcomeScreen->installEventFilter(this);
// Setup tab bar
m_tabWidget = new QTabWidget(this);
m_tabWidget->setTabsClosable(true);
m_tabWidget->setElideMode(Qt::ElideRight);
m_tabWidget->tabBar()->hide();
m_tabWidget->setDocumentMode(true);
m_tabWidget->setMovable(true);
m_tabWidget->setAcceptDrops(true);
m_tabWidget->tabBar()->installEventFilter(this);
m_centralStackedWidget->addWidget(m_tabWidget);
connect(m_tabWidget, &QTabWidget::currentChanged, this, &Shell::setActiveTab);
connect(m_tabWidget, &QTabWidget::tabCloseRequested, this, &Shell::closeTab);
connect(m_tabWidget->tabBar(), &QTabBar::tabMoved, this, &Shell::moveTabData);
m_sidebar = new Sidebar;
m_sidebar->setObjectName(QStringLiteral("okular_sidebar"));
m_sidebar->setContextMenuPolicy(Qt::ActionsContextMenu);
m_sidebar->setWindowTitle(i18n("Sidebar"));
connect(m_sidebar, &QDockWidget::visibilityChanged, this, [this](bool visible) {
// sync sidebar visibility with the m_showSidebarAction only if welcome screen is hidden
if (m_showSidebarAction && m_centralStackedWidget->currentWidget() != m_welcomeScreen) {
m_showSidebarAction->setChecked(visible);
}
});
addDockWidget(Qt::LeftDockWidgetArea, m_sidebar);
// then, setup our actions
setupActions();
connect(QCoreApplication::instance(), &QCoreApplication::aboutToQuit, this, &QObject::deleteLater);
// and integrate the part's GUI with the shell's
setupGUI(Keys | ToolBar | Save);
// NOTE : apply default sidebar width only after calling setupGUI(...)
resizeDocks({m_sidebar}, {200}, Qt::Horizontal);
m_tabs.append(TabState(firstPart));
m_tabWidget->addTab(firstPart->widget(), QString()); // triggers setActiveTab that calls createGUI( part )
connectPart(firstPart);
readSettings();
m_unique = ShellUtils::unique(serializedOptions);
#if HAVE_DBUS
if (m_unique) {
m_unique = QDBusConnection::sessionBus().registerService(QStringLiteral("org.kde.okular"));
if (!m_unique) {
KMessageBox::information(this, i18n("There is already a unique Okular instance running. This instance won't be the unique one."));
}
} else {
QString serviceName = QStringLiteral("org.kde.okular-") + QString::number(qApp->applicationPid());
QDBusConnection::sessionBus().registerService(serviceName);
}
if (ShellUtils::noRaise(serializedOptions)) {
setAttribute(Qt::WA_ShowWithoutActivating);
}
{
const QString editorCmd = ShellUtils::editorCmd(serializedOptions);
if (!editorCmd.isEmpty()) {
QMetaObject::invokeMethod(firstPart, "setEditorCmd", Q_ARG(QString, editorCmd));
}
}
QDBusConnection::sessionBus().registerObject(QStringLiteral("/okularshell"), this, QDBusConnection::ExportScriptableSlots);
#endif // HAVE_DBUS
// Make sure that the welcome scren is visible on startup.
showWelcomeScreen();
} else {
m_isValid = false;
KMessageBox::error(this, i18n("Unable to find the Okular component."));
}
connect(guiFactory(), &KXMLGUIFactory::shortcutsSaved, this, &Shell::reloadAllXML);
}
void Shell::reloadAllXML()
{
for (const TabState &tab : qAsConst(m_tabs)) {
tab.part->reloadXML();
}
}
void Shell::keyPressEvent(QKeyEvent *e)
{
if (e->key() == Qt::Key_Escape && window()->isFullScreen()) {
setFullScreen(false);
}
}
bool Shell::eventFilter(QObject *obj, QEvent *event)
{
QDragMoveEvent *dmEvent = dynamic_cast<QDragMoveEvent *>(event);
if (dmEvent) {
bool accept = dmEvent->mimeData()->hasUrls();
event->setAccepted(accept);
return accept;
}
QDropEvent *dEvent = dynamic_cast<QDropEvent *>(event);
if (dEvent) {
const QList<QUrl> list = KUrlMimeData::urlsFromMimeData(dEvent->mimeData());
handleDroppedUrls(list);
dEvent->setAccepted(true);
return true;
}
// Handle middle button click events on the tab bar
if (obj == m_tabWidget->tabBar() && event->type() == QEvent::MouseButtonRelease) {
QMouseEvent *mEvent = static_cast<QMouseEvent *>(event);
if (mEvent->button() == Qt::MiddleButton) {
int tabIndex = m_tabWidget->tabBar()->tabAt(mEvent->pos());
if (tabIndex != -1) {
closeTab(tabIndex);
return true;
}
}
}
return KParts::MainWindow::eventFilter(obj, event);
}
bool Shell::isValid() const
{
return m_isValid;
}
void Shell::showOpenRecentMenu()
{
m_recent->menu()->popup(QCursor::pos());
}
Shell::~Shell()
{
if (!m_tabs.empty()) {
writeSettings();
for (const TabState &tab : qAsConst(m_tabs)) {
tab.part->closeUrl(false);
}
m_tabs.clear();
}
#if HAVE_DBUS
if (m_unique) {
QDBusConnection::sessionBus().unregisterService(QStringLiteral("org.kde.okular"));
}
#endif // HAVE_DBUS
delete m_tabWidget;
}
// Open a new document if we have space for it
// This can hang if called on a unique instance and openUrl pops a messageBox
bool Shell::openDocument(const QUrl &url, const QString &serializedOptions)
{
if (m_tabs.size() <= 0) {
return false;
}
hideWelcomeScreen();
KParts::ReadWritePart *const part = m_tabs[0].part;
// Return false if we can't open new tabs and the only part is occupied
if (!qobject_cast<Okular::ViewerInterface *>(part)->openNewFilesInTabs() && !part->url().isEmpty() && !ShellUtils::unique(serializedOptions)) {
return false;
}
openUrl(url, serializedOptions);
return true;
}
bool Shell::openDocument(const QString &urlString, const QString &serializedOptions)
{
return openDocument(QUrl(urlString), serializedOptions);
}
bool Shell::canOpenDocs(int numDocs, int desktop)
{
if (m_tabs.size() <= 0 || numDocs <= 0 || m_unique) {
return false;
}
KParts::ReadWritePart *const part = m_tabs[0].part;
const bool allowTabs = qobject_cast<Okular::ViewerInterface *>(part)->openNewFilesInTabs();
if (!allowTabs && (numDocs > 1 || !part->url().isEmpty())) {
return false;
}
const KWindowInfo winfo(window()->effectiveWinId(), KWindowSystem::WMDesktop);
if (winfo.desktop() != desktop) {
return false;
}
return true;
}
void Shell::openUrl(const QUrl &url, const QString &serializedOptions)
{
hideWelcomeScreen();
const int activeTab = m_tabWidget->currentIndex();
if (activeTab < m_tabs.size()) {
KParts::ReadWritePart *const activePart = m_tabs[activeTab].part;
if (!activePart->url().isEmpty()) {
if (m_unique) {
applyOptionsToPart(activePart, serializedOptions);
activePart->openUrl(url);
} else {
if (qobject_cast<Okular::ViewerInterface *>(activePart)->openNewFilesInTabs()) {
openNewTab(url, serializedOptions);
} else {
Shell *newShell = new Shell(serializedOptions);
newShell->show();
newShell->openUrl(url, serializedOptions);
}
}
} else {
m_tabWidget->setTabText(activeTab, url.fileName());
m_tabWidget->setTabToolTip(activeTab, url.fileName());
applyOptionsToPart(activePart, serializedOptions);
bool openOk = activePart->openUrl(url);
const bool isstdin = url.fileName() == QLatin1String("-") || url.scheme() == QLatin1String("fd");
if (!isstdin) {
if (openOk) {
#ifdef WITH_KACTIVITIES
if (!m_activityResource) {
m_activityResource = new KActivities::ResourceInstance(window()->winId(), this);
}
m_activityResource->setUri(url);
#endif
m_recent->addUrl(url);
} else {
m_recent->removeUrl(url);
closeTab(activeTab);
}
}
}
}
}
void Shell::closeUrl()
{
closeTab(m_tabWidget->currentIndex());
// When closing the current tab two things can happen:
// * the focus was on the tab
// * the focus was somewhere in the toolbar
// we don't have other places that accept focus
// * If it was on the tab, logic says it should go back to the next current tab
// * If it was on the toolbar, we could leave it there, but since we redo the menus/toolbars for the new tab, it gets kind of lost
// so it's easier to set it to the next current tab which also makes sense as consistency
if (m_tabWidget->count() >= 0) {
KParts::ReadWritePart *const newPart = m_tabs[m_tabWidget->currentIndex()].part;
newPart->widget()->setFocus();
}
}
void Shell::readSettings()
{
m_recent->loadEntries(KSharedConfig::openConfig()->group("Recent Files"));
m_recent->setEnabled(true); // force enabling
const KConfigGroup group = KSharedConfig::openConfig()->group("Desktop Entry");
bool fullScreen = group.readEntry("FullScreen", false);
setFullScreen(fullScreen);
if (fullScreen) {
m_menuBarWasShown = group.readEntry(shouldShowMenuBarComingFromFullScreen, true);
m_toolBarWasShown = group.readEntry(shouldShowToolBarComingFromFullScreen, true);
}
const KConfigGroup sidebarGroup = KSharedConfig::openConfig()->group("General");
m_sidebar->setVisible(sidebarGroup.readEntry(SIDEBAR_VISIBLE_KEY, true));
m_sidebar->setLocked(sidebarGroup.readEntry(SIDEBAR_LOCKED_KEY, true));
m_showSidebarAction->setChecked(m_sidebar->isVisibleTo(this));
m_lockSidebarAction->setChecked(m_sidebar->isLocked());
}
void Shell::writeSettings()
{
saveRecents();
KConfigGroup sidebarGroup = KSharedConfig::openConfig()->group("General");
sidebarGroup.writeEntry(SIDEBAR_LOCKED_KEY, m_sidebar->isLocked());
// NOTE : Consider whether the m_showSidebarAction is checked, because
// the sidebar can be forcibly hidden if the welcome screen is displayed
sidebarGroup.writeEntry(SIDEBAR_VISIBLE_KEY, m_sidebar->isVisibleTo(this) || m_showSidebarAction->isChecked());
KConfigGroup group = KSharedConfig::openConfig()->group("Desktop Entry");
group.writeEntry("FullScreen", m_fullScreenAction->isChecked());
if (m_fullScreenAction->isChecked()) {
group.writeEntry(shouldShowMenuBarComingFromFullScreen, m_menuBarWasShown);
group.writeEntry(shouldShowToolBarComingFromFullScreen, m_toolBarWasShown);
}
KSharedConfig::openConfig()->sync();
}
void Shell::saveRecents()
{
m_recent->saveEntries(KSharedConfig::openConfig()->group("Recent Files"));
}
void Shell::setupActions()
{
KStandardAction::open(this, SLOT(fileOpen()), actionCollection());
m_recent = KStandardAction::openRecent(this, SLOT(openUrl(QUrl)), actionCollection());
m_recent->setToolBarMode(KRecentFilesAction::MenuMode);
connect(m_recent, &QAction::triggered, this, &Shell::showOpenRecentMenu);
connect(m_recent, &KRecentFilesAction::recentListCleared, this, &Shell::refreshRecentsOnWelcomeScreen);
connect(m_welcomeScreen, &WelcomeScreen::forgetAllRecents, m_recent, &KRecentFilesAction::clear);
m_recent->setToolTip(i18n("Click to open a file\nClick and hold to open a recent file"));
m_recent->setWhatsThis(i18n("<b>Click</b> to open a file or <b>Click and hold</b> to select a recent file"));
m_printAction = KStandardAction::print(this, SLOT(print()), actionCollection());
m_printAction->setEnabled(false);
m_closeAction = KStandardAction::close(this, SLOT(closeUrl()), actionCollection());
m_closeAction->setEnabled(false);
KStandardAction::quit(this, SLOT(close()), actionCollection());
setStandardToolBarMenuEnabled(true);
m_showMenuBarAction = KStandardAction::showMenubar(this, SLOT(slotShowMenubar()), actionCollection());
m_fullScreenAction = KStandardAction::fullScreen(this, SLOT(slotUpdateFullScreen()), this, actionCollection());
m_nextTabAction = actionCollection()->addAction(QStringLiteral("tab-next"));
m_nextTabAction->setText(i18n("Next Tab"));
actionCollection()->setDefaultShortcuts(m_nextTabAction, KStandardShortcut::tabNext());
m_nextTabAction->setEnabled(false);
connect(m_nextTabAction, &QAction::triggered, this, &Shell::activateNextTab);
m_prevTabAction = actionCollection()->addAction(QStringLiteral("tab-previous"));
m_prevTabAction->setText(i18n("Previous Tab"));
actionCollection()->setDefaultShortcuts(m_prevTabAction, KStandardShortcut::tabPrev());
m_prevTabAction->setEnabled(false);
connect(m_prevTabAction, &QAction::triggered, this, &Shell::activatePrevTab);
m_undoCloseTab = actionCollection()->addAction(QStringLiteral("undo-close-tab"));
m_undoCloseTab->setText(i18n("Undo close tab"));
actionCollection()->setDefaultShortcut(m_undoCloseTab, QKeySequence(Qt::CTRL | Qt::SHIFT | Qt::Key_T));
m_undoCloseTab->setIcon(QIcon::fromTheme(QStringLiteral("edit-undo")));
m_undoCloseTab->setEnabled(false);
connect(m_undoCloseTab, &QAction::triggered, this, &Shell::undoCloseTab);
m_lockSidebarAction = actionCollection()->addAction(QStringLiteral("okular_lock_sidebar"));
m_lockSidebarAction->setCheckable(true);
m_lockSidebarAction->setIcon(QIcon::fromTheme(QStringLiteral("lock")));
m_lockSidebarAction->setText(i18n("Lock Sidebar"));
connect(m_lockSidebarAction, &QAction::triggered, m_sidebar, &Sidebar::setLocked);
m_sidebar->addAction(m_lockSidebarAction);
}
void Shell::saveProperties(KConfigGroup &group)
{
if (!m_isValid) { // part couldn't be loaded, nothing to save
return;
}
// Gather lists of settings to preserve
QStringList urls;
for (const TabState &tab : qAsConst(m_tabs)) {
urls.append(tab.part->url().url());
}
group.writePathEntry(SESSION_URL_KEY, urls);
group.writeEntry(SESSION_TAB_KEY, m_tabWidget->currentIndex());
}
void Shell::readProperties(const KConfigGroup &group)
{
// Reopen documents based on saved settings
QStringList urls = group.readPathEntry(SESSION_URL_KEY, QStringList());
int desiredTab = group.readEntry<int>(SESSION_TAB_KEY, 0);
while (!urls.isEmpty()) {
openUrl(QUrl(urls.takeFirst()));
}
if (desiredTab < m_tabs.size()) {
setActiveTab(desiredTab);
}
}
void Shell::fileOpen()
{
// this slot is called whenever the File->Open menu is selected,
// the Open shortcut is pressed (usually CTRL+O) or the Open toolbar
// button is clicked
const int activeTab = m_tabWidget->currentIndex();
if (!m_fileformatsscanned) {
const KDocumentViewer *const doc = qobject_cast<KDocumentViewer *>(m_tabs[activeTab].part);
Q_ASSERT(doc);
m_fileformats = doc->supportedMimeTypes();
m_fileformatsscanned = true;
}
QUrl startDir;
const KParts::ReadWritePart *const curPart = m_tabs[activeTab].part;
if (curPart->url().isLocalFile()) {
startDir = KIO::upUrl(curPart->url());
}
QPointer<QFileDialog> dlg(new QFileDialog(this));
dlg->setDirectoryUrl(startDir);
dlg->setAcceptMode(QFileDialog::AcceptOpen);
dlg->setOption(QFileDialog::HideNameFilterDetails, true);
dlg->setFileMode(QFileDialog::ExistingFiles); // Allow selection of more than one file
QMimeDatabase mimeDatabase;
// Unfortunately non Plasma file dialogs don't support the "All supported files" when using
// setMimeTypeFilters instead of setNameFilters, so for those use setNameFilters which is a bit
// worse because doesn't show you pdf files named bla.blo when you say "show me the pdf files", but
// that's solvable by choosing "All Files" and it's not that common while it's more convenient to
// only get shown the files that the application can open by default instead of all of them
const bool useMimeTypeFilters = qgetenv("XDG_CURRENT_DESKTOP").toLower() == "kde";
if (useMimeTypeFilters) {
QStringList mimetypes;
for (const QString &mimeName : qAsConst(m_fileformats)) {
QMimeType mimeType = mimeDatabase.mimeTypeForName(mimeName);
mimetypes << mimeType.name();
}
mimetypes.prepend(QStringLiteral("application/octet-stream"));
dlg->setMimeTypeFilters(mimetypes);
} else {
QSet<QString> globPatterns;
QMap<QString, QStringList> namedGlobs;
for (const QString &mimeName : qAsConst(m_fileformats)) {
QMimeType mimeType = mimeDatabase.mimeTypeForName(mimeName);
const QStringList globs(mimeType.globPatterns());
if (globs.isEmpty()) {
continue;
}
globPatterns.unite(QSet<QString>(globs.begin(), globs.end()));
namedGlobs[mimeType.comment()].append(globs);
}
QStringList namePatterns;
for (auto it = namedGlobs.cbegin(); it != namedGlobs.cend(); ++it) {
namePatterns.append(it.key() + QLatin1String(" (") + it.value().join(QLatin1Char(' ')) + QLatin1Char(')'));
}
const QStringList allGlobPatterns = globPatterns.values();
namePatterns.prepend(i18n("All files (*)"));
namePatterns.prepend(i18n("All supported files (%1)", allGlobPatterns.join(QLatin1Char(' '))));
dlg->setNameFilters(namePatterns);
}
dlg->setWindowTitle(i18n("Open Document"));
if (dlg->exec() && dlg) {
const QList<QUrl> urlList = dlg->selectedUrls();
for (const QUrl &url : urlList) {
openUrl(url);
}
}
if (dlg) {
delete dlg.data();
}
}
void Shell::tryRaise(const QString &startupId)
{
if (KWindowSystem::isPlatformWayland()) {
KWindowSystem::setCurrentXdgActivationToken(startupId);
} else if (KWindowSystem::isPlatformX11()) {
KStartupInfo::setNewStartupId(window()->windowHandle(), startupId.toUtf8());
}
KWindowSystem::activateWindow(window()->windowHandle());
}
// only called when starting the program
void Shell::setFullScreen(bool useFullScreen)
{
if (useFullScreen) {
setWindowState(windowState() | Qt::WindowFullScreen); // set
} else {
setWindowState(windowState() & ~Qt::WindowFullScreen); // reset
}
}
void Shell::setCaption(const QString &caption)
{
bool modified = false;
const int activeTab = m_tabWidget->currentIndex();
if (activeTab >= 0 && activeTab < m_tabs.size()) {
KParts::ReadWritePart *const activePart = m_tabs[activeTab].part;
QString tabCaption = activePart->url().fileName();
if (activePart->isModified()) {
modified = true;
if (!tabCaption.isEmpty()) {
tabCaption.append(QStringLiteral(" *"));
}
}
m_tabWidget->setTabText(activeTab, tabCaption);
}
setCaption(caption, modified);
}
void Shell::showEvent(QShowEvent *e)
{
if (!menuBar()->isNativeMenuBar() && m_showMenuBarAction) {
m_showMenuBarAction->setChecked(menuBar()->isVisible());
}
KParts::MainWindow::showEvent(e);
}
void Shell::slotUpdateFullScreen()
{
if (m_fullScreenAction->isChecked()) {
m_menuBarWasShown = !menuBar()->isHidden();
menuBar()->hide();
m_toolBarWasShown = !toolBar()->isHidden();
toolBar()->hide();
KToggleFullScreenAction::setFullScreen(this, true);
} else {
if (m_menuBarWasShown) {
menuBar()->show();
}
if (m_toolBarWasShown) {
toolBar()->show();
}
KToggleFullScreenAction::setFullScreen(this, false);
}
}
void Shell::slotShowMenubar()
{
if (menuBar()->isHidden()) {
menuBar()->show();
} else {
menuBar()->hide();
}
}
QSize Shell::sizeHint() const
{
const QSize baseSize = QApplication::primaryScreen()->availableSize() * 0.6;
// Set an arbitrary yet sensible sane minimum size for very small screens;
// for example we don't want people using 1366x768 screens to get a tiny
// default window size of 820 x 460 which will elide most of the toolbar buttons.
return baseSize.expandedTo(QSize(1000, 700));
}
bool Shell::queryClose()
{
if (m_tabs.count() > 1) {
const QString dontAskAgainName = QStringLiteral("ShowTabWarning");
KMessageBox::ButtonCode dummy = KMessageBox::Yes;
if (shouldBeShownYesNo(dontAskAgainName, dummy)) {
QDialog *dialog = new QDialog(this);
dialog->setWindowTitle(i18n("Confirm Close"));
QDialogButtonBox *buttonBox = new QDialogButtonBox(dialog);
buttonBox->setStandardButtons(QDialogButtonBox::Yes | QDialogButtonBox::No);
KGuiItem::assign(buttonBox->button(QDialogButtonBox::Yes), KGuiItem(i18n("Close Tabs"), QStringLiteral("tab-close")));
KGuiItem::assign(buttonBox->button(QDialogButtonBox::No), KStandardGuiItem::cancel());
bool checkboxResult = true;
const int result = KMessageBox::createKMessageBox(dialog,
buttonBox,
QMessageBox::Question,
i18n("You are about to close %1 tabs. Are you sure you want to continue?", m_tabs.count()),
QStringList(),
i18n("Warn me when I attempt to close multiple tabs"),
&checkboxResult,
KMessageBox::Notify);
if (!checkboxResult) {
saveDontShowAgainYesNo(dontAskAgainName, dummy);
}
if (result != QDialogButtonBox::Yes) {
return false;
}
}
}
for (int i = 0; i < m_tabs.size(); ++i) {
KParts::ReadWritePart *const part = m_tabs[i].part;
// To resolve confusion about multiple modified docs, switch to relevant tab
if (part->isModified()) {
setActiveTab(i);
}
if (!part->queryClose()) {
return false;
}
}
return true;
}
void Shell::setActiveTab(int tab)
{
if (m_showSidebarAction) {
m_showSidebarAction->disconnect();
}
m_tabWidget->setCurrentIndex(tab);
// NOTE : createGUI(...) breaks the visibility of the sidebar, so we need
// to save and restore it
const bool isSidebarVisible = m_sidebar->isVisible();
createGUI(m_tabs[tab].part);
m_sidebar->setVisible(isSidebarVisible);
// dock KPart's sidebar if new and make it current
Okular::ViewerInterface *iPart = qobject_cast<Okular::ViewerInterface *>(m_tabs[tab].part);
Q_ASSERT(iPart);
QWidget *sideContainer = iPart->getSideContainer();
if (m_sidebar->indexOf(sideContainer) == -1) {
m_sidebar->addWidget(sideContainer);
if (m_sidebar->maximumWidth() > sideContainer->maximumWidth()) {
m_sidebar->setMaximumWidth(sideContainer->maximumWidth());
}
}
m_sidebar->setCurrentWidget(sideContainer);
m_showSidebarAction = m_tabs[tab].part->actionCollection()->action(QStringLiteral("show_leftpanel"));
Q_ASSERT(m_showSidebarAction);
m_showSidebarAction->disconnect();
m_showSidebarAction->setChecked(m_sidebar->isVisibleTo(this));
connect(m_showSidebarAction, &QAction::triggered, m_sidebar, &Sidebar::setVisible);
m_printAction->setEnabled(m_tabs[tab].printEnabled);
m_closeAction->setEnabled(m_tabs[tab].closeEnabled);
}
void Shell::closeTab(int tab)
{
KParts::ReadWritePart *const part = m_tabs[tab].part;
QUrl url = part->url();
bool closeSuccess = part->closeUrl();
if (closeSuccess && m_tabs.count() > 1) {
if (part->factory()) {
part->factory()->removeClient(part);
}
part->disconnect();
Okular::ViewerInterface *iPart = qobject_cast<Okular::ViewerInterface *>(m_tabs[tab].part);
Q_ASSERT(iPart);
QWidget *sideContainer = iPart->getSideContainer();
m_sidebar->removeWidget(sideContainer);
connect(part, &QObject::destroyed, sideContainer, &QObject::deleteLater);
part->deleteLater();
m_tabs.removeAt(tab);
m_tabWidget->removeTab(tab);
m_undoCloseTab->setEnabled(true);
m_closedTabUrls.append(url);
if (m_tabWidget->count() == 1) {
m_tabWidget->tabBar()->hide();
m_nextTabAction->setEnabled(false);
m_prevTabAction->setEnabled(false);
}
} else if (closeSuccess && m_tabs.count() == 1) {
// Show welcome screen when the last tab is closed.
showWelcomeScreen();
}
}
void Shell::openNewTab(const QUrl &url, const QString &serializedOptions)
{
const int previousActiveTab = m_tabWidget->currentIndex();
KParts::ReadWritePart *const activePart = m_tabs[previousActiveTab].part;
hideWelcomeScreen();
bool activateTabIfAlreadyOpen;
QMetaObject::invokeMethod(activePart, "activateTabIfAlreadyOpenFile", Q_RETURN_ARG(bool, activateTabIfAlreadyOpen));
if (activateTabIfAlreadyOpen) {
const int tabIndex = findTabIndex(url);
if (tabIndex >= 0) {
setActiveTab(tabIndex);
m_recent->addUrl(url);
return;
}
}
// Tabs are hidden when there's only one, so show it
if (m_tabs.size() == 1) {
m_tabWidget->tabBar()->show();
m_nextTabAction->setEnabled(true);
m_prevTabAction->setEnabled(true);
}
const int newIndex = m_tabs.size();
// Make new part
m_tabs.append(TabState(m_partFactory->create<KParts::ReadWritePart>(this)));
connectPart(m_tabs[newIndex].part);
// Update GUI
KParts::ReadWritePart *const part = m_tabs[newIndex].part;
m_tabWidget->addTab(part->widget(), url.fileName());
m_tabWidget->setTabToolTip(newIndex, url.fileName());
applyOptionsToPart(part, serializedOptions);
setActiveTab(m_tabs.size() - 1);
if (part->openUrl(url)) {
m_recent->addUrl(url);
} else {
setActiveTab(previousActiveTab);
closeTab(m_tabs.size() - 1);
m_recent->removeUrl(url);
}
}
void Shell::applyOptionsToPart(QObject *part, const QString &serializedOptions)
{
KDocumentViewer *const doc = qobject_cast<KDocumentViewer *>(part);
const QString find = ShellUtils::find(serializedOptions);
if (ShellUtils::startInPresentation(serializedOptions)) {
doc->startPresentation();
}
if (ShellUtils::showPrintDialog(serializedOptions)) {
QMetaObject::invokeMethod(part, "enableStartWithPrint");
}
if (ShellUtils::showPrintDialogAndExit(serializedOptions)) {
QMetaObject::invokeMethod(part, "enableExitAfterPrint");
}
if (!find.isEmpty()) {
QMetaObject::invokeMethod(part, "enableStartWithFind", Q_ARG(QString, find));
}
}
void Shell::connectPart(const KParts::ReadWritePart *part)
{
// We're abusing the fact we know the part is our part here
connect(this, SIGNAL(moveSplitter(int)), part, SLOT(moveSplitter(int))); // clazy:exclude=old-style-connect
connect(part, SIGNAL(enablePrintAction(bool)), this, SLOT(setPrintEnabled(bool))); // clazy:exclude=old-style-connect
connect(part, SIGNAL(enableCloseAction(bool)), this, SLOT(setCloseEnabled(bool))); // clazy:exclude=old-style-connect
connect(part, SIGNAL(mimeTypeChanged(QMimeType)), this, SLOT(setTabIcon(QMimeType))); // clazy:exclude=old-style-connect
connect(part, SIGNAL(urlsDropped(QList<QUrl>)), this, SLOT(handleDroppedUrls(QList<QUrl>))); // clazy:exclude=old-style-connect
// clang-format off
// Otherwise the QSize,QSize gets turned into QSize, QSize that is not normalized signals and is slightly slower
connect(part, SIGNAL(fitWindowToPage(QSize,QSize)), this, SLOT(slotFitWindowToPage(QSize,QSize))); // clazy:exclude=old-style-connect
// clang-format on
}
void Shell::print()
{
QMetaObject::invokeMethod(m_tabs[m_tabWidget->currentIndex()].part, "slotPrint");
}
void Shell::setPrintEnabled(bool enabled)
{
int i = findTabIndex(sender());
if (i != -1) {
m_tabs[i].printEnabled = enabled;
if (i == m_tabWidget->currentIndex()) {
m_printAction->setEnabled(enabled);
}
}
}
void Shell::setCloseEnabled(bool enabled)
{
int i = findTabIndex(sender());
if (i != -1) {
m_tabs[i].closeEnabled = enabled;
if (i == m_tabWidget->currentIndex()) {
m_closeAction->setEnabled(enabled);
}
}
}
void Shell::activateNextTab()
{
if (m_tabs.size() < 2) {
return;
}
const int activeTab = m_tabWidget->currentIndex();
const int nextTab = (activeTab == m_tabs.size() - 1) ? 0 : activeTab + 1;
setActiveTab(nextTab);
}
void Shell::activatePrevTab()
{
if (m_tabs.size() < 2) {
return;
}
const int activeTab = m_tabWidget->currentIndex();
const int prevTab = (activeTab == 0) ? m_tabs.size() - 1 : activeTab - 1;
setActiveTab(prevTab);
}
void Shell::undoCloseTab()
{
if (m_closedTabUrls.isEmpty()) {
return;
}
const QUrl lastTabUrl = m_closedTabUrls.takeLast();
if (m_closedTabUrls.isEmpty()) {
m_undoCloseTab->setEnabled(false);
}
openUrl(lastTabUrl);
}
void Shell::setTabIcon(const QMimeType &mimeType)
{
int i = findTabIndex(sender());
if (i != -1) {
m_tabWidget->setTabIcon(i, QIcon::fromTheme(mimeType.iconName()));
}
}
int Shell::findTabIndex(QObject *sender) const
{
for (int i = 0; i < m_tabs.size(); ++i) {
if (m_tabs[i].part == sender) {
return i;
}
}
return -1;
}
int Shell::findTabIndex(const QUrl &url) const
{
auto it = std::find_if(m_tabs.begin(), m_tabs.end(), [&url](const TabState state) { return state.part->url() == url; });
return (it != m_tabs.end()) ? std::distance(m_tabs.begin(), it) : -1;
}
void Shell::handleDroppedUrls(const QList<QUrl> &urls)
{
for (const QUrl &url : urls) {
openUrl(url);
}
}
void Shell::moveTabData(int from, int to)
{
m_tabs.move(from, to);
}
void Shell::slotFitWindowToPage(const QSize pageViewSize, const QSize pageSize)
{
const int xOffset = pageViewSize.width() - pageSize.width();
const int yOffset = pageViewSize.height() - pageSize.height();
showNormal();
resize(width() - xOffset, height() - yOffset);
Q_EMIT moveSplitter(pageSize.width());
}
void Shell::hideWelcomeScreen()
{
m_sidebar->setVisible(m_showSidebarAction->isChecked());
m_centralStackedWidget->setCurrentWidget(m_tabWidget);
m_showSidebarAction->setEnabled(true);
}
void Shell::showWelcomeScreen()
{
m_showSidebarAction->setEnabled(false);
m_centralStackedWidget->setCurrentWidget(m_welcomeScreen);
m_sidebar->setVisible(false);
refreshRecentsOnWelcomeScreen();
}
void Shell::refreshRecentsOnWelcomeScreen()
{
saveRecents();
m_welcomeScreen->loadRecents();
}
void Shell::forgetRecentItem(QUrl const &url)
{
if (m_recent != nullptr) {
m_recent->removeUrl(url);
saveRecents();
refreshRecentsOnWelcomeScreen();
}
}
#include "shell.moc"
/* kate: replace-tabs on; indent-width 4; */