mirror of
https://github.com/clementine-player/Clementine
synced 2025-01-04 14:00:29 +01:00
370 lines
12 KiB
C++
370 lines
12 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
|
|
** All rights reserved.
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
**
|
|
** This file is part of a Qt Solutions component.
|
|
**
|
|
** Commercial Usage
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
** accordance with the Qt Solutions Commercial License Agreement provided
|
|
** with the Software or, alternatively, in accordance with the terms
|
|
** contained in a written agreement between you and Nokia.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** In addition, as a special exception, Nokia gives you certain
|
|
** additional rights. These rights are described in the Nokia Qt LGPL
|
|
** Exception version 1.1, included in the file LGPL_EXCEPTION.txt in this
|
|
** package.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3.0 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU General Public License version 3.0 requirements will be
|
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
|
**
|
|
** Please note Third Party Software included with Qt Solutions may impose
|
|
** additional restrictions and it is the user's responsibility to ensure
|
|
** that they have met the licensing requirements of the GPL, LGPL, or Qt
|
|
** Solutions Commercial license and the relevant license of the Third
|
|
** Party Software they are using.
|
|
**
|
|
** If you are unsure which license is appropriate for your use, please
|
|
** contact Nokia at qt-info@nokia.com.
|
|
**
|
|
****************************************************************************/
|
|
|
|
|
|
#include "qtsingleapplication.h"
|
|
#include "qtlocalpeer.h"
|
|
#include <QtGui/QWidget>
|
|
|
|
|
|
/*!
|
|
\class QtSingleApplication qtsingleapplication.h
|
|
\brief The QtSingleApplication class provides an API to detect and
|
|
communicate with running instances of an application.
|
|
|
|
This class allows you to create applications where only one
|
|
instance should be running at a time. I.e., if the user tries to
|
|
launch another instance, the already running instance will be
|
|
activated instead. Another usecase is a client-server system,
|
|
where the first started instance will assume the role of server,
|
|
and the later instances will act as clients of that server.
|
|
|
|
By default, the full path of the executable file is used to
|
|
determine whether two processes are instances of the same
|
|
application. You can also provide an explicit identifier string
|
|
that will be compared instead.
|
|
|
|
The application should create the QtSingleApplication object early
|
|
in the startup phase, and call isRunning() or sendMessage() to
|
|
find out if another instance of this application is already
|
|
running. Startup parameters (e.g. the name of the file the user
|
|
wanted this new instance to open) can be passed to the running
|
|
instance in the sendMessage() function.
|
|
|
|
If isRunning() or sendMessage() returns false, it means that no
|
|
other instance is running, and this instance has assumed the role
|
|
as the running instance. The application should continue with the
|
|
initialization of the application user interface before entering
|
|
the event loop with exec(), as normal. The messageReceived()
|
|
signal will be emitted when the application receives messages from
|
|
another instance of the same application.
|
|
|
|
If isRunning() or sendMessage() returns true, another instance is
|
|
already running, and the application should terminate or enter
|
|
client mode.
|
|
|
|
If a message is received it might be helpful to the user to raise
|
|
the application so that it becomes visible. To facilitate this,
|
|
QtSingleApplication provides the setActivationWindow() function
|
|
and the activateWindow() slot.
|
|
|
|
Here's an example that shows how to convert an existing
|
|
application to use QtSingleApplication. It is very simple and does
|
|
not make use of all QtSingleApplication's functionality (see the
|
|
examples for that).
|
|
|
|
\code
|
|
// Original
|
|
int main(int argc, char **argv)
|
|
{
|
|
QApplication app(argc, argv);
|
|
|
|
MyMainWidget mmw;
|
|
|
|
mmw.show();
|
|
return app.exec();
|
|
}
|
|
|
|
// Single instance
|
|
int main(int argc, char **argv)
|
|
{
|
|
QtSingleApplication app(argc, argv);
|
|
|
|
if (app.isRunning())
|
|
return 0;
|
|
|
|
MyMainWidget mmw;
|
|
|
|
app.setActivationWindow(&mmw);
|
|
|
|
mmw.show();
|
|
return app.exec();
|
|
}
|
|
\endcode
|
|
|
|
Once this QtSingleApplication instance is destroyed(for example,
|
|
when the user quits), when the user next attempts to run the
|
|
application this instance will not, of course, be encountered. The
|
|
next instance to call isRunning() or sendMessage() will assume the
|
|
role as the new running instance.
|
|
|
|
For console (non-GUI) applications, QtSingleCoreApplication may be
|
|
used instead of this class, to avoid the dependency on the QtGui
|
|
library.
|
|
|
|
\sa QtSingleCoreApplication
|
|
*/
|
|
|
|
|
|
void QtSingleApplication::sysInit(const QString &appId)
|
|
{
|
|
actWin = 0;
|
|
peer = new QtLocalPeer(this, appId);
|
|
connect(peer, SIGNAL(messageReceived(const QString&)), SIGNAL(messageReceived(const QString&)));
|
|
connect(peer, SIGNAL(messageReceived(const QByteArray&)), SIGNAL(messageReceived(const QByteArray&)));
|
|
connect(peer, SIGNAL(messageReceived(const char*)), SIGNAL(messageReceived(const char*)));
|
|
}
|
|
|
|
|
|
/*!
|
|
Creates a QtSingleApplication object. The application identifier
|
|
will be QCoreApplication::applicationFilePath(). \a argc, \a
|
|
argv, and \a GUIenabled are passed on to the QAppliation constructor.
|
|
|
|
If you are creating a console application (i.e. setting \a
|
|
GUIenabled to false), you may consider using
|
|
QtSingleCoreApplication instead.
|
|
*/
|
|
|
|
QtSingleApplication::QtSingleApplication(int &argc, char **argv, bool GUIenabled)
|
|
: QApplication(argc, argv, GUIenabled)
|
|
{
|
|
sysInit();
|
|
}
|
|
|
|
|
|
/*!
|
|
Creates a QtSingleApplication object with the application
|
|
identifier \a appId. \a argc and \a argv are passed on to the
|
|
QAppliation constructor.
|
|
*/
|
|
|
|
QtSingleApplication::QtSingleApplication(const QString &appId, int &argc, char **argv)
|
|
: QApplication(argc, argv)
|
|
{
|
|
sysInit(appId);
|
|
}
|
|
|
|
|
|
/*!
|
|
Creates a QtSingleApplication object. The application identifier
|
|
will be QCoreApplication::applicationFilePath(). \a argc, \a
|
|
argv, and \a type are passed on to the QAppliation constructor.
|
|
*/
|
|
QtSingleApplication::QtSingleApplication(int &argc, char **argv, Type type)
|
|
: QApplication(argc, argv, type)
|
|
{
|
|
sysInit();
|
|
}
|
|
|
|
|
|
#if defined(Q_WS_X11)
|
|
/*!
|
|
Special constructor for X11, ref. the documentation of
|
|
QApplication's corresponding constructor. The application identifier
|
|
will be QCoreApplication::applicationFilePath(). \a dpy, \a visual,
|
|
and \a cmap are passed on to the QApplication constructor.
|
|
*/
|
|
QtSingleApplication::QtSingleApplication(Display* dpy, Qt::HANDLE visual, Qt::HANDLE cmap)
|
|
: QApplication(dpy, visual, cmap)
|
|
{
|
|
sysInit();
|
|
}
|
|
|
|
/*!
|
|
Special constructor for X11, ref. the documentation of
|
|
QApplication's corresponding constructor. The application identifier
|
|
will be QCoreApplication::applicationFilePath(). \a dpy, \a argc, \a
|
|
argv, \a visual, and \a cmap are passed on to the QApplication
|
|
constructor.
|
|
*/
|
|
QtSingleApplication::QtSingleApplication(Display *dpy, int &argc, char **argv, Qt::HANDLE visual, Qt::HANDLE cmap)
|
|
: QApplication(dpy, argc, argv, visual, cmap)
|
|
{
|
|
sysInit();
|
|
}
|
|
|
|
/*!
|
|
Special constructor for X11, ref. the documentation of
|
|
QApplication's corresponding constructor. The application identifier
|
|
will be \a appId. \a dpy, \a argc, \a
|
|
argv, \a visual, and \a cmap are passed on to the QApplication
|
|
constructor.
|
|
*/
|
|
QtSingleApplication::QtSingleApplication(Display* dpy, const QString &appId, int argc, char **argv, Qt::HANDLE visual, Qt::HANDLE cmap)
|
|
: QApplication(dpy, argc, argv, visual, cmap)
|
|
{
|
|
sysInit(appId);
|
|
}
|
|
#endif
|
|
|
|
|
|
/*!
|
|
Returns true if another instance of this application is running;
|
|
otherwise false.
|
|
|
|
This function does not find instances of this application that are
|
|
being run by a different user (on Windows: that are running in
|
|
another session).
|
|
|
|
\sa sendMessage()
|
|
*/
|
|
|
|
bool QtSingleApplication::isRunning()
|
|
{
|
|
return peer->isClient();
|
|
}
|
|
|
|
|
|
/*!
|
|
Tries to send the text \a message to the currently running
|
|
instance. The QtSingleApplication object in the running instance
|
|
will emit the messageReceived() signal when it receives the
|
|
message.
|
|
|
|
This function returns true if the message has been sent to, and
|
|
processed by, the current instance. If there is no instance
|
|
currently running, or if the running instance fails to process the
|
|
message within \a timeout milliseconds, this function return false.
|
|
|
|
\sa isRunning(), messageReceived()
|
|
*/
|
|
bool QtSingleApplication::sendMessage(const QString &message, int timeout)
|
|
{
|
|
return peer->sendMessage(message, timeout);
|
|
}
|
|
|
|
bool QtSingleApplication::sendMessage(const QByteArray &message, int timeout)
|
|
{
|
|
return peer->sendMessage(message, timeout);
|
|
}
|
|
|
|
bool QtSingleApplication::sendMessage(const char* message, int timeout)
|
|
{
|
|
return peer->sendMessage(message, timeout);
|
|
}
|
|
|
|
|
|
/*!
|
|
Returns the application identifier. Two processes with the same
|
|
identifier will be regarded as instances of the same application.
|
|
*/
|
|
QString QtSingleApplication::id() const
|
|
{
|
|
return peer->applicationId();
|
|
}
|
|
|
|
|
|
/*!
|
|
Sets the activation window of this application to \a aw. The
|
|
activation window is the widget that will be activated by
|
|
activateWindow(). This is typically the application's main window.
|
|
|
|
If \a activateOnMessage is true (the default), the window will be
|
|
activated automatically every time a message is received, just prior
|
|
to the messageReceived() signal being emitted.
|
|
|
|
\sa activateWindow(), messageReceived()
|
|
*/
|
|
|
|
void QtSingleApplication::setActivationWindow(QWidget* aw, bool activateOnMessage)
|
|
{
|
|
actWin = aw;
|
|
if (activateOnMessage) {
|
|
connect(peer, SIGNAL(messageReceived(const QString&)), this, SLOT(activateWindow()));
|
|
connect(peer, SIGNAL(messageReceived(const QByteArray&)), this, SLOT(activateWindow()));
|
|
connect(peer, SIGNAL(messageReceived(const char*)), this, SLOT(activateWindow()));
|
|
}
|
|
else {
|
|
disconnect(peer, SIGNAL(messageReceived(const QString&)), this, SLOT(activateWindow()));
|
|
disconnect(peer, SIGNAL(messageReceived(const QByteArray&)), this, SLOT(activateWindow()));
|
|
disconnect(peer, SIGNAL(messageReceived(const char*)), this, SLOT(activateWindow()));
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
Returns the applications activation window if one has been set by
|
|
calling setActivationWindow(), otherwise returns 0.
|
|
|
|
\sa setActivationWindow()
|
|
*/
|
|
QWidget* QtSingleApplication::activationWindow() const
|
|
{
|
|
return actWin;
|
|
}
|
|
|
|
|
|
/*!
|
|
De-minimizes, raises, and activates this application's activation window.
|
|
This function does nothing if no activation window has been set.
|
|
|
|
This is a convenience function to show the user that this
|
|
application instance has been activated when he has tried to start
|
|
another instance.
|
|
|
|
This function should typically be called in response to the
|
|
messageReceived() signal. By default, that will happen
|
|
automatically, if an activation window has been set.
|
|
|
|
\sa setActivationWindow(), messageReceived(), initialize()
|
|
*/
|
|
void QtSingleApplication::activateWindow()
|
|
{
|
|
if (actWin) {
|
|
actWin->setWindowState(actWin->windowState() & ~Qt::WindowMinimized);
|
|
actWin->raise();
|
|
actWin->activateWindow();
|
|
}
|
|
}
|
|
|
|
|
|
/*!
|
|
\fn void QtSingleApplication::messageReceived(const QString& message)
|
|
|
|
This signal is emitted when the current instance receives a \a
|
|
message from another instance of this application.
|
|
|
|
\sa sendMessage(), setActivationWindow(), activateWindow()
|
|
*/
|
|
|
|
|
|
/*!
|
|
\fn void QtSingleApplication::initialize(bool dummy = true)
|
|
|
|
\obsolete
|
|
*/
|