1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
/****************************************************************************
**
** Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies).
** Contact: Qt Software Information (qt-info.com)
**
** This file is part of a Qt Solutions component.
**
** Commercial Usage
** Licensees holding valid Qt Solutions 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.
**
** If you are unsure which license is appropriate for your use, please
** contact the sales department at qt-sales.com.
**
****************************************************************************/
#include "qtsingleapplication.h"
#include <QtGui/QWidget>
class QtSingletonPrivate
{
public:
QString id;
};
/*!
\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 that cannot have
multiple instances running on the same machine for the same user.
To use the QtSingleApplication class you must provide an ID string
that it unique on the system you run the application on. Typical
IDs are the name of the application and the application vendor, or
a string representation of a \link QUuid UUID\endlink.
The application should create the QtSingleApplication object very
early in the startup phase, and try to send a message or call
isRunning() to find out if an instance of this application is
already running.
If an instance is already running, this application instance
should terminate. Otherwise the application should call
initialize() immediately, and continue with the initialization of
the application user interface before entering the event loop with
exec(). The messageReceived() signal will be emitted when the
application receives messages from another instance of the same
application.
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 us 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("MySingleInstance", argc, argv);
if (app.sendMessage("Do I exist?"))
return 0;
app.initialize();
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.
*/
/*!
Creates a QtSingleApplication object with the identifier \a id. \a
argc, \a argv and \a type are passed on to the QAppliation
constructor.
There can only be one QtSingleApplication object(and since there
can only be one QApplication object you do not need to create
another QApplication object yourself).
\warning On X11 type can not be QApplication::Tty.
*/
QtSingleApplication::QtSingleApplication(const QString &id, int &argc, char **argv, bool useGui)
: QApplication(argc, argv, useGui)
{
d = new QtSingletonPrivate;
d->id = id;
actWin = 0;
sysInit();
}
#ifdef Q_WS_X11
/*!
Creates a QtSingleApplication object, given an already open display
\a dpy. Uses the identifier \a id. \a argc and \a argv are
passed on to the QAppliation constructor. If \a visual and \a colormap
are non-zero, the application will use those as the default Visual and
Colormap contexts.
There can only be one QtSingleApplication object(and since there
can only be one QApplication object you do not need to create
another QApplication object yourself).
\warning Qt only supports TrueColor visuals at depths higher than 8
bits-per-pixel.
This is available only on X11.
*/
/*
QtSingleApplication::QtSingleApplication(Display* dpy, const QString &id, int argc, char **argv,
Qt::HANDLE visual, Qt::HANDLE colormap)
: QApplication(dpy, argc, argv, visual, colormap)
{
d = new QtSingletonPrivate;
d->id = id;
actWin = 0;
sysInit();
}
*/
#endif // Q_WS_X11
/*!
Destroys the object, freeing all allocated resources.
If the same application is started again it will not find this
instance.
*/
QtSingleApplication::~QtSingleApplication()
{
sysCleanup();
delete d;
}
/*!
Returns the identifier of this singleton object.
*/
QString QtSingleApplication::id() const
{
return d->id;
}
/*!
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.
\sa activateWindow(), messageReceived()
*/
void QtSingleApplication::setActivationWindow(QWidget* aw)
{
actWin = aw;
}
/*!
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. initialize() will connect that
signal to this slot by default.
\sa setActivationWindow(), messageReceived(), initialize()
*/
void QtSingleApplication::activateWindow()
{
if (actWin) {
actWin->setWindowState(actWin->windowState() & ~Qt::WindowMinimized);
actWin->raise();
actWin->activateWindow();
}
}
/*! \fn bool QtSingleApplication::isRunning() const
Returns true if another instance of this application has called
initialize(); otherwise returns false.
This function does not find instances of this application that are
being run by a different user.
\sa initialize()
*/
/*!
\fn void QtSingleApplication::initialize(bool activate)
Once this function has been called, this application instance
becomes "visible" to other instances. This means that if another
instance is started(by the same user), and calls isRunning(), the
isRunning() function will return true to that other instance,
which should then quit, leaving this instance to continue.
If \a activate is true (the default) the messageReceived() signal
will be connected to the activateWindow() slot.
*/
/*!
\fn bool QtSingleApplication::sendMessage(const QString& message, int timeout)
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.
Note that on X11 systems the \a timeout parameter is ignored.
\sa messageReceived()
*/
/*!
\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.
This signal is typically connected to the activateWindow()
slot.
\sa activateWindow(), initialize()
*/