/[svn]/qsampler/trunk/src/qsampler.cpp
ViewVC logotype

Annotation of /qsampler/trunk/src/qsampler.cpp

Parent Directory Parent Directory | Revision Log Revision Log


Revision 3001 - (hide annotations) (download)
Sat Oct 8 22:08:39 2016 UTC (7 years, 5 months ago) by capela
File size: 13317 byte(s)
- Fixed a potential crash on the singleton/unique application
  instance setup.
1 capela 2074 // qsampler.cpp
2     //
3     /****************************************************************************
4 capela 2987 Copyright (C) 2004-2016, rncbc aka Rui Nuno Capela. All rights reserved.
5 capela 2074 Copyright (C) 2007, 2008 Christian Schoenebeck
6    
7     This program is free software; you can redistribute it and/or
8     modify it under the terms of the GNU General Public License
9     as published by the Free Software Foundation; either version 2
10     of the License, or (at your option) any later version.
11    
12     This program is distributed in the hope that it will be useful,
13     but WITHOUT ANY WARRANTY; without even the implied warranty of
14     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15     GNU General Public License for more details.
16    
17     You should have received a copy of the GNU General Public License along
18     with this program; if not, write to the Free Software Foundation, Inc.,
19     51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
20    
21     *****************************************************************************/
22    
23     #include "qsamplerAbout.h"
24     #include "qsamplerOptions.h"
25     #include "qsamplerMainForm.h"
26    
27     #include <QApplication>
28     #include <QLibraryInfo>
29     #include <QTranslator>
30     #include <QLocale>
31 capela 2459
32 capela 2074 #if defined(__APPLE__) // Toshi Nagata 20080105
33     #include <QDir>
34     #endif
35    
36 capela 2998 #ifndef CONFIG_PREFIX
37     #define CONFIG_PREFIX "/usr/local"
38     #endif
39    
40     #ifndef CONFIG_DATADIR
41     #define CONFIG_DATADIR CONFIG_PREFIX "/share"
42     #endif
43    
44 persson 2210 #if WIN32
45     #define RELATIVE_LOCALE_DIR "/share/locale"
46     #elif defined(__APPLE__)
47     #define RELATIVE_LOCALE_DIR "/../Resources"
48     #endif
49 capela 2074
50 persson 2210
51 capela 2074 //-------------------------------------------------------------------------
52     // Singleton application instance stuff (Qt/X11 only atm.)
53     //
54    
55 capela 2839 #if QT_VERSION < 0x050000
56 capela 2074 #if defined(Q_WS_X11)
57 capela 2839 #define CONFIG_X11
58     #endif
59     #else
60     #if defined(QT_X11EXTRAS_LIB)
61     #define CONFIG_X11
62     #endif
63     #endif
64 capela 2074
65 capela 2839
66     #ifdef CONFIG_X11
67     #ifdef CONFIG_XUNIQUE
68    
69 capela 2074 #include <QX11Info>
70    
71     #include <X11/Xatom.h>
72     #include <X11/Xlib.h>
73    
74 capela 2839 #define QSAMPLER_XUNIQUE "qsamplerApplication"
75 capela 2074
76 capela 2839 #if QT_VERSION >= 0x050100
77    
78     #include <xcb/xcb.h>
79     #include <xcb/xproto.h>
80    
81     #include <QAbstractNativeEventFilter>
82    
83     class qsamplerApplication;
84    
85     class qsamplerXcbEventFilter : public QAbstractNativeEventFilter
86     {
87     public:
88    
89     // Constructor.
90     qsamplerXcbEventFilter(qsamplerApplication *pApp)
91     : QAbstractNativeEventFilter(), m_pApp(pApp) {}
92    
93     // XCB event filter (virtual processor).
94     bool nativeEventFilter(const QByteArray& eventType, void *message, long *);
95    
96     private:
97    
98     // Instance variable.
99     qsamplerApplication *m_pApp;
100     };
101    
102 capela 2074 #endif
103    
104 capela 2839 #endif // CONFIG_XUNIQUE
105     #endif // CONFIG_X11
106    
107    
108 capela 2074 class qsamplerApplication : public QApplication
109     {
110     public:
111    
112     // Constructor.
113     qsamplerApplication(int& argc, char **argv) : QApplication(argc, argv),
114 capela 2839 m_pQtTranslator(0), m_pMyTranslator(0), m_pWidget(0)
115 capela 2074 {
116     // Load translation support.
117     QLocale loc;
118     if (loc.language() != QLocale::C) {
119     // Try own Qt translation...
120     m_pQtTranslator = new QTranslator(this);
121     QString sLocName = "qt_" + loc.name();
122     QString sLocPath = QLibraryInfo::location(QLibraryInfo::TranslationsPath);
123     if (m_pQtTranslator->load(sLocName, sLocPath)) {
124     QApplication::installTranslator(m_pQtTranslator);
125     } else {
126 capela 2631 #ifdef RELATIVE_LOCALE_DIR
127 persson 2210 sLocPath = QApplication::applicationDirPath() + RELATIVE_LOCALE_DIR;
128 persson 2144 if (m_pQtTranslator->load(sLocName, sLocPath)) {
129     QApplication::installTranslator(m_pQtTranslator);
130     } else {
131 capela 2631 #endif
132 capela 2074 delete m_pQtTranslator;
133     m_pQtTranslator = 0;
134 capela 2631 #ifdef CONFIG_DEBUG
135 capela 2074 qWarning("Warning: no translation found for '%s' locale: %s/%s.qm",
136     loc.name().toUtf8().constData(),
137     sLocPath.toUtf8().constData(),
138     sLocName.toUtf8().constData());
139 capela 2631 #endif
140     #ifdef RELATIVE_LOCALE_DIR
141 persson 2144 }
142 capela 2631 #endif
143 capela 2074 }
144     // Try own application translation...
145     m_pMyTranslator = new QTranslator(this);
146     sLocName = "qsampler_" + loc.name();
147     if (m_pMyTranslator->load(sLocName, sLocPath)) {
148     QApplication::installTranslator(m_pMyTranslator);
149     } else {
150 capela 2631 #ifdef RELATIVE_LOCALE_DIR
151 persson 2210 sLocPath = QApplication::applicationDirPath() + RELATIVE_LOCALE_DIR;
152 capela 2631 #else
153     sLocPath = CONFIG_DATADIR "/qsampler/translations";
154     #endif
155 capela 2074 if (m_pMyTranslator->load(sLocName, sLocPath)) {
156     QApplication::installTranslator(m_pMyTranslator);
157     } else {
158     delete m_pMyTranslator;
159     m_pMyTranslator = 0;
160 capela 2631 #ifdef CONFIG_DEBUG
161 capela 2074 qWarning("Warning: no translation found for '%s' locale: %s/%s.qm",
162     loc.name().toUtf8().constData(),
163     sLocPath.toUtf8().constData(),
164     sLocName.toUtf8().constData());
165 capela 2631 #endif
166 capela 2074 }
167     }
168     }
169 capela 2839 #ifdef CONFIG_X11
170     #ifdef CONFIG_XUNIQUE
171     // Instance uniqueness initialization...
172 capela 2074 m_pDisplay = QX11Info::display();
173     m_aUnique = XInternAtom(m_pDisplay, QSAMPLER_XUNIQUE, false);
174     XGrabServer(m_pDisplay);
175     m_wOwner = XGetSelectionOwner(m_pDisplay, m_aUnique);
176     XUngrabServer(m_pDisplay);
177 capela 2839 #if QT_VERSION >= 0x050100
178     m_pXcbEventFilter = new qsamplerXcbEventFilter(this);
179     installNativeEventFilter(m_pXcbEventFilter);
180 capela 2074 #endif
181 capela 2839 #endif // CONFIG_XUNIQUE
182     #endif // CONFIG_X11
183 capela 2074 }
184    
185     // Destructor.
186     ~qsamplerApplication()
187     {
188 capela 2839 #ifdef CONFIG_X11
189     #ifdef CONFIG_XUNIQUE
190     #if QT_VERSION >= 0x050100
191     removeNativeEventFilter(m_pXcbEventFilter);
192     delete m_pXcbEventFilter;
193     #endif
194     #endif // CONFIG_XUNIQUE
195     #endif // CONFIG_X11
196 capela 2074 if (m_pMyTranslator) delete m_pMyTranslator;
197     if (m_pQtTranslator) delete m_pQtTranslator;
198     }
199    
200     // Main application widget accessors.
201     void setMainWidget(QWidget *pWidget)
202     {
203     m_pWidget = pWidget;
204 capela 2839 #ifdef CONFIG_X11
205     #ifdef CONFIG_XUNIQUE
206 capela 2074 m_wOwner = m_pWidget->winId();
207 capela 3001 if (m_pDisplay && m_wOwner) {
208     XGrabServer(m_pDisplay);
209     XSetSelectionOwner(m_pDisplay, m_aUnique, m_wOwner, CurrentTime);
210     XUngrabServer(m_pDisplay);
211     }
212 capela 2839 #endif // CONFIG_XUNIQUE
213     #endif // CONFIG_X11
214 capela 2074 }
215    
216     QWidget *mainWidget() const { return m_pWidget; }
217    
218     // Check if another instance is running,
219     // and raise its proper main widget...
220     bool setup()
221     {
222 capela 2839 #ifdef CONFIG_X11
223     #ifdef CONFIG_XUNIQUE
224 capela 3001 if (m_pDisplay && m_wOwner != None) {
225 capela 2074 // First, notify any freedesktop.org WM
226     // that we're about to show the main widget...
227     Screen *pScreen = XDefaultScreenOfDisplay(m_pDisplay);
228     int iScreen = XScreenNumberOfScreen(pScreen);
229     XEvent ev;
230     memset(&ev, 0, sizeof(ev));
231     ev.xclient.type = ClientMessage;
232     ev.xclient.display = m_pDisplay;
233     ev.xclient.window = m_wOwner;
234     ev.xclient.message_type = XInternAtom(m_pDisplay, "_NET_ACTIVE_WINDOW", false);
235     ev.xclient.format = 32;
236     ev.xclient.data.l[0] = 0; // Source indication.
237     ev.xclient.data.l[1] = 0; // Timestamp.
238     ev.xclient.data.l[2] = 0; // Requestor's currently active window (none)
239     ev.xclient.data.l[3] = 0;
240     ev.xclient.data.l[4] = 0;
241     XSelectInput(m_pDisplay, m_wOwner, StructureNotifyMask);
242     XSendEvent(m_pDisplay, RootWindow(m_pDisplay, iScreen), false,
243     (SubstructureNotifyMask | SubstructureRedirectMask), &ev);
244     XSync(m_pDisplay, false);
245     XRaiseWindow(m_pDisplay, m_wOwner);
246     // And then, let it get caught on destination
247 capela 2839 // by QApplication::native/x11EventFilter...
248     const QByteArray value = QSAMPLER_XUNIQUE;
249 capela 2074 XChangeProperty(
250     m_pDisplay,
251     m_wOwner,
252     m_aUnique,
253     m_aUnique, 8,
254     PropModeReplace,
255     (unsigned char *) value.data(),
256     value.length());
257     // Done.
258     return true;
259     }
260 capela 2839 #endif // CONFIG_XUNIQUE
261     #endif // CONFIG_X11
262 capela 2074 return false;
263     }
264    
265 capela 2839 #ifdef CONFIG_X11
266     #ifdef CONFIG_XUNIQUE
267     void x11PropertyNotify(Window w)
268 capela 2074 {
269 capela 3001 if (m_pDisplay && m_pWidget && m_wOwner == w) {
270 capela 2074 // Always check whether our property-flag is still around...
271     Atom aType;
272     int iFormat = 0;
273     unsigned long iItems = 0;
274     unsigned long iAfter = 0;
275     unsigned char *pData = 0;
276     if (XGetWindowProperty(
277     m_pDisplay,
278     m_wOwner,
279     m_aUnique,
280     0, 1024,
281     false,
282     m_aUnique,
283     &aType,
284     &iFormat,
285     &iItems,
286     &iAfter,
287     &pData) == Success
288     && aType == m_aUnique && iItems > 0 && iAfter == 0) {
289     // Avoid repeating it-self...
290     XDeleteProperty(m_pDisplay, m_wOwner, m_aUnique);
291     // Just make it always shows up fine...
292     m_pWidget->show();
293     m_pWidget->raise();
294     m_pWidget->activateWindow();
295     }
296     // Free any left-overs...
297     if (iItems > 0 && pData)
298     XFree(pData);
299     }
300 capela 2839 }
301     #if QT_VERSION < 0x050000
302     bool x11EventFilter(XEvent *pEv)
303     {
304     if (pEv->type == PropertyNotify
305     && pEv->xproperty.state == PropertyNewValue)
306     x11PropertyNotify(pEv->xproperty.window);
307 capela 2074 return QApplication::x11EventFilter(pEv);
308     }
309     #endif
310 capela 2839 #endif // CONFIG_XUNIQUE
311     #endif // CONFIG_X11
312 capela 2074
313     private:
314    
315     // Translation support.
316     QTranslator *m_pQtTranslator;
317     QTranslator *m_pMyTranslator;
318    
319     // Instance variables.
320     QWidget *m_pWidget;
321    
322 capela 2839 #ifdef CONFIG_X11
323     #ifdef CONFIG_XUNIQUE
324 capela 2074 Display *m_pDisplay;
325     Atom m_aUnique;
326     Window m_wOwner;
327 capela 2839 #if QT_VERSION >= 0x050100
328     qsamplerXcbEventFilter *m_pXcbEventFilter;
329 capela 2074 #endif
330 capela 2839 #endif // CONFIG_XUNIQUE
331     #endif // CONFIG_X11
332 capela 2074 };
333    
334    
335 capela 2839 #ifdef CONFIG_X11
336     #ifdef CONFIG_XUNIQUE
337     #if QT_VERSION >= 0x050100
338     // XCB Event filter (virtual processor).
339     bool qsamplerXcbEventFilter::nativeEventFilter (
340     const QByteArray& eventType, void *message, long * )
341     {
342     if (eventType == "xcb_generic_event_t") {
343     xcb_property_notify_event_t *pEv
344     = static_cast<xcb_property_notify_event_t *> (message);
345     if ((pEv->response_type & ~0x80) == XCB_PROPERTY_NOTIFY
346     && pEv->state == XCB_PROPERTY_NEW_VALUE)
347     m_pApp->x11PropertyNotify(pEv->window);
348     }
349     return false;
350     }
351     #endif
352     #endif // CONFIG_XUNIQUE
353     #endif // CONFIG_X11
354    
355    
356 capela 2074 //-------------------------------------------------------------------------
357     // stacktrace - Signal crash handler.
358     //
359    
360     #ifdef CONFIG_STACKTRACE
361     #if defined(__GNUC__) && defined(Q_OS_LINUX)
362    
363     #include <stdio.h>
364     #include <errno.h>
365     #include <signal.h>
366 capela 2341 #include <unistd.h>
367 capela 2074 #include <sys/wait.h>
368    
369     void stacktrace ( int signo )
370     {
371     pid_t pid;
372     int rc;
373     int status = 0;
374     char cmd[80];
375    
376     // Reinstall default handler; prevent race conditions...
377     signal(signo, SIG_DFL);
378    
379     static const char *shell = "/bin/sh";
380 capela 2181 static const char *format = "gdb -q --batch --pid=%d"
381     " --eval-command='thread apply all bt'";
382 capela 2074
383     snprintf(cmd, sizeof(cmd), format, (int) getpid());
384    
385     pid = fork();
386    
387     // Fork failure!
388     if (pid < 0)
389     return;
390    
391     // Fork child...
392     if (pid == 0) {
393     execl(shell, shell, "-c", cmd, NULL);
394     _exit(1);
395     return;
396     }
397    
398     // Parent here: wait for child to terminate...
399     do { rc = waitpid(pid, &status, 0); }
400     while ((rc < 0) && (errno == EINTR));
401    
402     // Dispatch any logging, if any...
403     QApplication::processEvents(QEventLoop::AllEvents, 3000);
404    
405     // Make sure everyone terminates...
406     kill(pid, SIGTERM);
407     _exit(1);
408     }
409    
410     #endif
411     #endif
412    
413    
414     //-------------------------------------------------------------------------
415     // main - The main program trunk.
416     //
417    
418     int main ( int argc, char **argv )
419     {
420     Q_INIT_RESOURCE(qsampler);
421     #ifdef CONFIG_STACKTRACE
422     #if defined(__GNUC__) && defined(Q_OS_LINUX)
423     signal(SIGILL, stacktrace);
424     signal(SIGFPE, stacktrace);
425     signal(SIGSEGV, stacktrace);
426     signal(SIGABRT, stacktrace);
427     signal(SIGBUS, stacktrace);
428     #endif
429     #endif
430     qsamplerApplication app(argc, argv);
431    
432     #if defined(__APPLE__) // Toshi Nagata 20080105
433     {
434     // Set the plugin path to @exetutable_path/../plugins
435     QDir dir(QApplication::applicationDirPath());
436     dir.cdUp(); // "Contents" directory
437     QApplication::setLibraryPaths(QStringList(dir.absolutePath() + "/plugins"));
438    
439     // Set the PATH environment variable to include @executable_path/../../..
440     dir.cdUp();
441     dir.cdUp();
442     QString path(getenv("PATH"));
443     path = dir.absolutePath() + ":" + path;
444     setenv("PATH", path.toUtf8().constData(), 1);
445     }
446     #endif
447    
448     // Construct default settings; override with command line arguments.
449     QSampler::Options options;
450     if (!options.parse_args(app.arguments())) {
451     app.quit();
452     return 1;
453     }
454    
455     // Have another instance running?
456     if (app.setup()) {
457     app.quit();
458     return 2;
459     }
460    
461     // Dark themes grayed/disabled color group fix...
462     QPalette pal(app.palette());
463     if (pal.base().color().value() < 0x7f) {
464 capela 2415 #if QT_VERSION >= 0x050000
465     const QColor& color = pal.window().color();
466     const int iGroups = int(QPalette::Active | QPalette::Inactive) + 1;
467     for (int i = 0; i < iGroups; ++i) {
468     const QPalette::ColorGroup group = QPalette::ColorGroup(i);
469     pal.setBrush(group, QPalette::Light, color.lighter(150));
470     pal.setBrush(group, QPalette::Midlight, color.lighter(120));
471     pal.setBrush(group, QPalette::Dark, color.darker(150));
472     pal.setBrush(group, QPalette::Mid, color.darker(120));
473     pal.setBrush(group, QPalette::Shadow, color.darker(200));
474     }
475     // pal.setColor(QPalette::Disabled, QPalette::ButtonText, pal.mid().color());
476     #endif
477 capela 2074 pal.setColorGroup(QPalette::Disabled,
478     pal.windowText().color().darker(),
479     pal.button(),
480     pal.light(),
481     pal.dark(),
482     pal.mid(),
483     pal.text().color().darker(),
484     pal.text().color().lighter(),
485     pal.base(),
486     pal.window());
487     app.setPalette(pal);
488     }
489    
490     // Set default base font...
491     if (options.iBaseFontSize > 0)
492 capela 2677 app.setFont(QFont(app.font().family(), options.iBaseFontSize));
493 capela 2074
494     // Construct, setup and show the main form.
495     QSampler::MainForm w;
496     w.setup(&options);
497     w.show();
498    
499     // Settle this one as application main widget...
500     app.setMainWidget(&w);
501    
502     // Register the quit signal/slot.
503     // app.connect(&app, SIGNAL(lastWindowClosed()), &app, SLOT(quit()));
504    
505     return app.exec();
506     }
507    
508    
509     // end of qsampler.cpp

  ViewVC Help
Powered by ViewVC