okular/shell/okular_main.cpp
2023-12-18 22:56:22 +00:00

247 lines
8.3 KiB
C++

/*
SPDX-FileCopyrightText: 2002 Wilco Greven <greven@kde.org>
SPDX-FileCopyrightText: 2003 Christophe Devriese <Christophe.Devriese@student.kuleuven.ac.be>
SPDX-FileCopyrightText: 2003 Laurent Montel <montel@kde.org>
SPDX-FileCopyrightText: 2003-2007 Albert Astals Cid <aacid@kde.org>
SPDX-FileCopyrightText: 2004 Andy Goossens <andygoossens@telenet.be>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "okular_main.h"
#include "aboutdata.h"
#include "shell.h"
#include "shellutils.h"
#include <KLocalizedString>
#include <KWindowSystem>
#include <QApplication>
#include <QMimeData>
#include <QTemporaryFile>
#include <QTextStream>
#include "config-okular.h"
#if HAVE_X11
#include <KX11Extras>
#include <private/qtx11extras_p.h>
#endif
#if HAVE_DBUS
#include <QDBusConnectionInterface>
#include <QDBusInterface>
#endif // HAVE_DBUS
#include <iostream>
static QString startupId()
{
QString result;
if (KWindowSystem::isPlatformWayland()) {
result = qEnvironmentVariable("XDG_ACTIVATION_TOKEN");
qunsetenv("XDG_ACTIVATION_TOKEN");
} else if (KWindowSystem::isPlatformX11()) {
#if HAVE_X11
result = QString::fromUtf8(QX11Info::nextStartupId());
#endif
}
return result;
}
static bool attachUniqueInstance(const QStringList &paths, const QString &serializedOptions)
{
#if HAVE_DBUS
if (!ShellUtils::unique(serializedOptions) || paths.count() != 1) {
return false;
}
QDBusInterface iface(QStringLiteral("org.kde.okular"), QStringLiteral("/okularshell"), QStringLiteral("org.kde.okular"));
if (!iface.isValid()) {
return false;
}
if (!ShellUtils::editorCmd(serializedOptions).isEmpty()) {
QString message =
i18n("You cannot set the editor command in an already running okular instance. Please disable the tabs and try again. Please note, that unique is also not supported when setting the editor command at the commandline.\n");
std::cerr << message.toStdString();
exit(1);
}
const QString page = ShellUtils::page(serializedOptions);
iface.call(QStringLiteral("openDocument"), ShellUtils::urlFromArg(paths[0], ShellUtils::qfileExistFunc(), page).url(), serializedOptions);
if (!ShellUtils::noRaise(serializedOptions)) {
iface.call(QStringLiteral("tryRaise"), startupId());
}
return true;
#else // HAVE_DBUS
return false;
#endif // HAVE_DBUS
}
// Ask an existing non-unique instance to open new tabs
static bool attachExistingInstance(const QStringList &paths, const QString &serializedOptions)
{
#if HAVE_DBUS
if (paths.count() < 1) {
return false;
}
// Don't try to attach to an existing instance with --print-and-exit because that would mean
// we're going to exit that other instance and that's just rude
if (ShellUtils::showPrintDialogAndExit(serializedOptions)) {
return false;
}
// If DBus isn't running, we can't attach to an existing instance.
auto *sessionInterface = QDBusConnection::sessionBus().interface();
if (!sessionInterface) {
return false;
}
const QStringList services = sessionInterface->registeredServiceNames().value();
// Don't match the service without trailing "-" (unique instance)
const QString pattern = QStringLiteral("org.kde.okular-");
const QString myPid = QString::number(qApp->applicationPid());
QScopedPointer<QDBusInterface> bestService;
#if HAVE_X11
const int desktop = KX11Extras::currentDesktop();
#else
const int desktop = 0;
#endif
// Select the first instance that isn't us (metric may change in future)
for (const QString &service : services) {
if (service.startsWith(pattern) && !service.endsWith(myPid)) {
bestService.reset(new QDBusInterface(service, QStringLiteral("/okularshell"), QStringLiteral("org.kde.okular")));
// Find a window that can handle our documents
const QDBusReply<bool> reply = bestService->call(QStringLiteral("canOpenDocs"), (int)paths.count(), desktop);
if (reply.isValid() && reply.value()) {
break;
}
bestService.reset();
}
}
if (!bestService) {
return false;
}
for (const QString &arg : paths) {
// Copy stdin to temporary file which can be opened by the existing
// window. The temp file is automatically deleted after it has been
// opened. Not sure if this behavior is safe on all platforms.
QScopedPointer<QTemporaryFile> tempFile;
QString path;
if (arg == QLatin1String("-")) {
tempFile.reset(new QTemporaryFile);
QFile stdinFile;
if (!tempFile->open() || !stdinFile.open(stdin, QIODevice::ReadOnly)) {
return false;
}
const size_t bufSize = 1024 * 1024;
QScopedPointer<char, QScopedPointerArrayDeleter<char>> buf(new char[bufSize]);
size_t bytes;
do {
bytes = stdinFile.read(buf.data(), bufSize);
tempFile->write(buf.data(), bytes);
} while (bytes != 0);
path = tempFile->fileName();
} else {
// Page only makes sense if we are opening one file
const QString page = ShellUtils::page(serializedOptions);
path = ShellUtils::urlFromArg(arg, ShellUtils::qfileExistFunc(), page).url();
}
// Returns false if it can't fit another document
const QDBusReply<bool> reply = bestService->call(QStringLiteral("openDocument"), path, serializedOptions);
if (!reply.isValid() || !reply.value()) {
return false;
}
}
if (!ShellUtils::editorCmd(serializedOptions).isEmpty()) {
QString message(
i18n("You cannot set the editor command in an already running okular instance. Please disable the tabs and try again. Please note, that unique is also not supported when setting the editor command at the commandline.\n"));
std::cerr << message.toStdString();
exit(1);
}
bestService->call(QStringLiteral("tryRaise"), startupId());
return true;
#else // HAVE_DBUS
return false;
#endif // HAVE_DBUS
}
namespace Okular
{
Status main(const QStringList &paths, const QString &serializedOptions)
{
if (ShellUtils::unique(serializedOptions) && paths.count() > 1) {
QTextStream stream(stderr);
stream << i18n("Error: Can't open more than one document with the --unique switch") << '\n';
return Error;
}
if (ShellUtils::startInPresentation(serializedOptions) && paths.count() > 1) {
QTextStream stream(stderr);
stream << i18n("Error: Can't open more than one document with the --presentation switch") << '\n';
return Error;
}
if (ShellUtils::showPrintDialog(serializedOptions) && paths.count() > 1) {
QTextStream stream(stderr);
stream << i18n("Error: Can't open more than one document with the --print switch") << '\n';
return Error;
}
if (!ShellUtils::page(serializedOptions).isEmpty() && paths.count() > 1) {
QTextStream stream(stderr);
stream << i18n("Error: Can't open more than one document with the --page switch") << '\n';
return Error;
}
if (!ShellUtils::find(serializedOptions).isEmpty() && paths.count() > 1) {
QTextStream stream(stderr);
stream << i18n("Error: Can't open more than one document with the --find switch") << '\n';
return Error;
}
// try to attach to existing session, unique or not
if (attachUniqueInstance(paths, serializedOptions) || attachExistingInstance(paths, serializedOptions)) {
return AttachedOtherProcess;
}
Shell *shell = new Shell(serializedOptions);
if (!shell->isValid()) {
return Error;
}
shell->show();
for (int i = 0; i < paths.count();) {
// Page only makes sense if we are opening one file
const QString page = ShellUtils::page(serializedOptions);
const QUrl url = ShellUtils::urlFromArg(paths[i], ShellUtils::qfileExistFunc(), page);
if (shell->openDocument(url, serializedOptions)) {
++i;
} else {
shell = new Shell(serializedOptions);
if (!shell->isValid()) {
return Error;
}
shell->show();
}
}
return Success;
}
}
/* kate: replace-tabs on; indent-width 4; */