2016-07-05 01:19:48 +08:00
|
|
|
// The MIT License (MIT)
|
|
|
|
//
|
2018-07-27 09:29:55 +08:00
|
|
|
// Copyright (c) Itay Grudev 2015 - 2018
|
2016-07-05 01:19:48 +08:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
2015-02-27 03:00:11 +08:00
|
|
|
#ifndef SINGLE_APPLICATION_H
|
|
|
|
#define SINGLE_APPLICATION_H
|
2016-05-05 04:05:59 +08:00
|
|
|
|
2016-03-05 02:15:13 +08:00
|
|
|
#include <QtCore/QtGlobal>
|
2016-08-10 09:42:46 +08:00
|
|
|
#include <QtNetwork/QLocalSocket>
|
2012-12-23 06:12:38 +08:00
|
|
|
|
2016-04-06 01:31:05 +08:00
|
|
|
#ifndef QAPPLICATION_CLASS
|
2016-05-10 00:05:38 +08:00
|
|
|
#define QAPPLICATION_CLASS QCoreApplication
|
2016-04-06 01:31:05 +08:00
|
|
|
#endif
|
|
|
|
|
2016-03-05 02:15:13 +08:00
|
|
|
#include QT_STRINGIFY(QAPPLICATION_CLASS)
|
2015-06-06 05:26:41 +08:00
|
|
|
|
|
|
|
class SingleApplicationPrivate;
|
2012-12-23 06:12:38 +08:00
|
|
|
|
|
|
|
/**
|
2019-08-24 03:57:28 +08:00
|
|
|
* @brief The SingleApplication class handles multiple instances of the same
|
2016-08-10 09:42:46 +08:00
|
|
|
* Application
|
|
|
|
* @see QCoreApplication
|
2012-12-23 06:12:38 +08:00
|
|
|
*/
|
2015-06-09 22:29:20 +08:00
|
|
|
class SingleApplication : public QAPPLICATION_CLASS
|
2012-12-23 06:12:38 +08:00
|
|
|
{
|
2016-03-05 02:15:13 +08:00
|
|
|
Q_OBJECT
|
|
|
|
|
2020-05-17 22:50:27 +08:00
|
|
|
using app_t = QAPPLICATION_CLASS;
|
2016-03-05 02:15:13 +08:00
|
|
|
|
2012-12-23 06:12:38 +08:00
|
|
|
public:
|
2016-08-10 09:42:46 +08:00
|
|
|
/**
|
|
|
|
* @brief Mode of operation of SingleApplication.
|
|
|
|
* Whether the block should be user-wide or system-wide and whether the
|
|
|
|
* primary instance should be notified when a secondary instance had been
|
|
|
|
* started.
|
|
|
|
* @note Operating system can restrict the shared memory blocks to the same
|
|
|
|
* user, in which case the User/System modes will have no effect and the
|
|
|
|
* block will be user wide.
|
|
|
|
* @enum
|
|
|
|
*/
|
|
|
|
enum Mode {
|
|
|
|
User = 1 << 0,
|
|
|
|
System = 1 << 1,
|
2016-11-27 01:40:47 +08:00
|
|
|
SecondaryNotification = 1 << 2,
|
|
|
|
ExcludeAppVersion = 1 << 3,
|
|
|
|
ExcludeAppPath = 1 << 4
|
2016-08-10 09:42:46 +08:00
|
|
|
};
|
|
|
|
Q_DECLARE_FLAGS(Options, Mode)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Intitializes a SingleApplication instance with argc command line
|
|
|
|
* arguments in argv
|
|
|
|
* @arg {int &} argc - Number of arguments in argv
|
|
|
|
* @arg {const char *[]} argv - Supplied command line arguments
|
|
|
|
* @arg {bool} allowSecondary - Whether to start the instance as secondary
|
|
|
|
* if there is already a primary instance.
|
|
|
|
* @arg {Mode} mode - Whether for the SingleApplication block to be applied
|
|
|
|
* User wide or System wide.
|
2019-08-24 03:57:28 +08:00
|
|
|
* @arg {int} timeout - Timeout to wait in milliseconds.
|
2016-08-10 09:42:46 +08:00
|
|
|
* @note argc and argv may be changed as Qt removes arguments that it
|
|
|
|
* recognizes
|
|
|
|
* @note Mode::SecondaryNotification only works if set on both the primary
|
|
|
|
* instance and the secondary instance.
|
|
|
|
* @note The timeout is just a hint for the maximum time of blocking
|
|
|
|
* operations. It does not guarantee that the SingleApplication
|
|
|
|
* initialisation will be completed in given time, though is a good hint.
|
|
|
|
* Usually 4*timeout would be the worst case (fail) scenario.
|
|
|
|
* @see See the corresponding QAPPLICATION_CLASS constructor for reference
|
|
|
|
*/
|
2020-12-22 01:07:36 +08:00
|
|
|
explicit SingleApplication( int &argc, char *argv[], bool allowSecondary = false, Options options = Mode::User, int timeout = 1000, const QString &userData = {} );
|
2020-05-17 22:50:27 +08:00
|
|
|
~SingleApplication() override;
|
2012-12-23 06:12:38 +08:00
|
|
|
|
2016-08-10 09:42:46 +08:00
|
|
|
/**
|
|
|
|
* @brief Returns if the instance is the primary instance
|
|
|
|
* @returns {bool}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
bool isPrimary() const;
|
2016-08-10 09:42:46 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns if the instance is a secondary instance
|
|
|
|
* @returns {bool}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
bool isSecondary() const;
|
2016-05-05 02:59:07 +08:00
|
|
|
|
2016-08-10 09:42:46 +08:00
|
|
|
/**
|
|
|
|
* @brief Returns a unique identifier for the current instance
|
2017-10-02 19:17:41 +08:00
|
|
|
* @returns {qint32}
|
2016-08-10 09:42:46 +08:00
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
quint32 instanceId() const;
|
2012-12-23 06:12:38 +08:00
|
|
|
|
2017-10-02 19:17:41 +08:00
|
|
|
/**
|
|
|
|
* @brief Returns the process ID (PID) of the primary instance
|
|
|
|
* @returns {qint64}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
qint64 primaryPid() const;
|
2017-10-02 19:17:41 +08:00
|
|
|
|
2020-03-03 09:22:54 +08:00
|
|
|
/**
|
|
|
|
* @brief Returns the username of the user running the primary instance
|
|
|
|
* @returns {QString}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
QString primaryUser() const;
|
2020-03-03 09:22:54 +08:00
|
|
|
|
2020-03-27 15:00:14 +08:00
|
|
|
/**
|
|
|
|
* @brief Returns the username of the current user
|
|
|
|
* @returns {QString}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
QString currentUser() const;
|
2020-03-27 15:00:14 +08:00
|
|
|
|
2016-08-10 09:42:46 +08:00
|
|
|
/**
|
|
|
|
* @brief Sends a message to the primary instance. Returns true on success.
|
|
|
|
* @param {int} timeout - Timeout for connecting
|
|
|
|
* @returns {bool}
|
|
|
|
* @note sendMessage() will return false if invoked from the primary
|
|
|
|
* instance.
|
|
|
|
*/
|
2020-05-17 22:50:27 +08:00
|
|
|
bool sendMessage( const QByteArray &message, int timeout = 100 );
|
2016-08-10 09:42:46 +08:00
|
|
|
|
2020-12-11 08:18:00 +08:00
|
|
|
/**
|
|
|
|
* @brief Get the set user data.
|
|
|
|
* @returns {QStringList}
|
|
|
|
*/
|
2021-01-25 06:16:37 +08:00
|
|
|
QStringList userData() const;
|
2020-12-11 08:18:00 +08:00
|
|
|
|
2016-08-10 09:42:46 +08:00
|
|
|
Q_SIGNALS:
|
|
|
|
void instanceStarted();
|
|
|
|
void receivedMessage( quint32 instanceId, QByteArray message );
|
2015-02-27 03:00:11 +08:00
|
|
|
|
2012-12-23 06:12:38 +08:00
|
|
|
private:
|
2016-03-05 02:15:13 +08:00
|
|
|
SingleApplicationPrivate *d_ptr;
|
2016-08-10 09:42:46 +08:00
|
|
|
Q_DECLARE_PRIVATE(SingleApplication)
|
2020-09-09 07:28:02 +08:00
|
|
|
void abortSafely();
|
2012-12-23 06:12:38 +08:00
|
|
|
};
|
|
|
|
|
2016-08-10 09:42:46 +08:00
|
|
|
Q_DECLARE_OPERATORS_FOR_FLAGS(SingleApplication::Options)
|
|
|
|
|
2015-02-27 03:00:11 +08:00
|
|
|
#endif // SINGLE_APPLICATION_H
|