OSDN Git Service

5d645857b038329bb18dfd28deaa281ce2b89085
[mutilities/MUtilities.git] / src / Startup.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // MuldeR's Utilities for Qt
3 // Copyright (C) 2004-2014 LoRd_MuldeR <MuldeR2@GMX.de>
4 //
5 // This library is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU Lesser General Public
7 // License as published by the Free Software Foundation; either
8 // version 2.1 of the License, or (at your option) any later version.
9 //
10 // This library is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13 // Lesser General Public License for more details.
14 //
15 // You should have received a copy of the GNU Lesser General Public
16 // License along with this library; if not, write to the Free Software
17 // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
18 //
19 // http://www.gnu.org/licenses/lgpl-2.1.txt
20 //////////////////////////////////////////////////////////////////////////////////
21
22 //MUtils
23 #include <MUtils/Startup.h>
24 #include <MUtils/OSSupport.h>
25 #include <MUtils/Terminal.h>
26 #include <MUtils/ErrorHandler.h>
27 #include <MUtils/Exception.h>
28
29 //Qt
30 #include <QApplication>
31 #include <QMutex>
32 #include <QStringList>
33 #include <QLibraryInfo>
34 #include <QTextCodec>
35 #include <QImageReader>
36 #include <QFont>
37 #include <QMessageBox>
38
39 ///////////////////////////////////////////////////////////////////////////////
40 // MESSAGE HANDLER
41 ///////////////////////////////////////////////////////////////////////////////
42
43 static void qt_message_handler(QtMsgType type, const char *msg)
44 {
45         if((!msg) || (!(msg[0])))
46         {
47                 return;
48         }
49
50         MUtils::Terminal::write(type, msg);
51
52         if((type == QtCriticalMsg) || (type == QtFatalMsg))
53         {
54                 MUtils::OS::fatal_exit(MUTILS_WCHR(QString::fromUtf8(msg)));
55         }
56 }
57
58 static bool qt_event_filter(void *message, long *result)
59 {
60         return MUtils::OS::handle_os_message(message, result);
61 }
62
63 ///////////////////////////////////////////////////////////////////////////////
64 // STARTUP FUNCTION
65 ///////////////////////////////////////////////////////////////////////////////
66
67 static int startup_main(int &argc, char **argv, MUtils::Startup::main_function_t *const entry_point)
68 {
69         qInstallMsgHandler(qt_message_handler);
70         MUtils::Terminal::setup(argc, argv, MUTILS_DEBUG);
71         return entry_point(argc, argv);
72 }
73
74 static int startup_helper(int &argc, char **argv, MUtils::Startup::main_function_t *const entry_point)
75 {
76         int iResult = -1;
77         try
78         {
79                 iResult = startup_main(argc, argv, entry_point);
80         }
81         catch(const std::exception &error)
82         {
83                 MUTILS_PRINT_ERROR("\nGURU MEDITATION !!!\n\nException error:\n%s\n", error.what());
84                 MUtils::OS::fatal_exit(L"Unhandeled C++ exception error, application will exit!");
85         }
86         catch(...)
87         {
88                 MUTILS_PRINT_ERROR("\nGURU MEDITATION !!!\n\nUnknown exception error!\n");
89                 MUtils::OS::fatal_exit(L"Unhandeled C++ exception error, application will exit!");
90         }
91         return iResult;
92 }
93
94 int MUtils::Startup::startup(int &argc, char **argv, main_function_t *const entry_point)
95 {
96         int iResult = -1;
97 #if (MUTILS_DEBUG)
98 #ifdef _MSC_VER
99         _CrtSetDbgFlag(_CRTDBG_CHECK_ALWAYS_DF || _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG));
100 #endif //_MSCVER
101         iResult = startup_main(argc, argv, entry_point);
102 #else //MUTILS_DEBUG
103 #ifdef _MSC_VER
104         __try
105         {
106                 MUtils::ErrorHandler::initialize();
107                 iResult = startup_helper(argc, argv, entry_point);
108         }
109         __except(1)
110         {
111                 MUTILS_PRINT_ERROR("\nGURU MEDITATION !!!\n\nUnhandeled structured exception error!\n");
112                 MUtils::OS::fatal_exit(L"Unhandeled structured exception error, application will exit!");
113         }
114 #else //_MSCVER
115         MUtils::ErrorHandler::initialize();
116         iResult = startup_helper(argc, argv, entry_point);
117 #endif //_MSCVER
118 #endif //MUTILS_DEBUG
119         return iResult;
120 }
121
122 ///////////////////////////////////////////////////////////////////////////////
123 // QT INITIALIZATION
124 ///////////////////////////////////////////////////////////////////////////////
125
126 static QMutex g_qt_lock;
127 static QScopedPointer<QApplication> g_application;
128
129 static const char *const g_imageformats[] = {"bmp", "png", "jpg", "gif", "ico", "xpm", "svg", NULL};
130
131 bool MUtils::Startup::init_qt(int &argc, char **argv, const QString &appName)
132 {
133         QMutexLocker lock(&g_qt_lock);
134         const QStringList &arguments = MUtils::OS::arguments();
135
136         //Don't initialized again, if done already
137         if(!g_application.isNull())
138         {
139                 return true;
140         }
141
142         //Extract executable name from argv[] array
143         QString executableName = QLatin1String("LameXP.exe");
144         if(arguments.count() > 0)
145         {
146                 static const char *delimiters = "\\/:?";
147                 executableName = arguments[0].trimmed();
148                 for(int i = 0; delimiters[i]; i++)
149                 {
150                         int temp = executableName.lastIndexOf(QChar(delimiters[i]));
151                         if(temp >= 0) executableName = executableName.mid(temp + 1);
152                 }
153                 executableName = executableName.trimmed();
154                 if(executableName.isEmpty())
155                 {
156                         executableName = QLatin1String("LameXP.exe");
157                 }
158         }
159
160         //Check Qt version
161 #ifdef QT_BUILD_KEY
162         qDebug("Using Qt v%s [%s], %s, %s", qVersion(), QLibraryInfo::buildDate().toString(Qt::ISODate).toLatin1().constData(), (qSharedBuild() ? "DLL" : "Static"), QLibraryInfo::buildKey().toLatin1().constData());
163         qDebug("Compiled with Qt v%s [%s], %s\n", QT_VERSION_STR, QT_PACKAGEDATE_STR, QT_BUILD_KEY);
164         if(_stricmp(qVersion(), QT_VERSION_STR))
165         {
166                 qFatal("%s", QApplication::tr("Executable '%1' requires Qt v%2, but found Qt v%3.").arg(executableName, QString::fromLatin1(QT_VERSION_STR), QString::fromLatin1(qVersion())).toLatin1().constData());
167                 return false;
168         }
169         if(QLibraryInfo::buildKey().compare(QString::fromLatin1(QT_BUILD_KEY), Qt::CaseInsensitive))
170         {
171                 qFatal("%s", QApplication::tr("Executable '%1' was built for Qt '%2', but found Qt '%3'.").arg(executableName, QString::fromLatin1(QT_BUILD_KEY), QLibraryInfo::buildKey()).toLatin1().constData());
172                 return false;
173         }
174 #else
175         qDebug("Using Qt v%s [%s], %s", qVersion(), QLibraryInfo::buildDate().toString(Qt::ISODate).toLatin1().constData(), (qSharedBuild() ? "DLL" : "Static"));
176         qDebug("Compiled with Qt v%s [%s]\n", QT_VERSION_STR, QT_PACKAGEDATE_STR);
177 #endif
178
179         //Check the Windows version
180         
181         const MUtils::OS::Version::os_version_t &osVersion = MUtils::OS::os_version();
182         if((osVersion.type != MUtils::OS::Version::OS_WINDOWS) || (osVersion < MUtils::OS::Version::WINDOWS_WINXP))
183         {
184                 qFatal("%s", QApplication::tr("Executable '%1' requires Windows XP or later.").arg(executableName).toLatin1().constData());
185         }
186
187         //Check whether we are running on a supported Windows version
188         if(const char *const friendlyName = MUtils::OS::os_friendly_name(osVersion))
189         {
190                 qDebug("Running on %s (NT v%u.%u).\n", friendlyName, osVersion.versionMajor, osVersion.versionMinor);
191         }
192         else
193         {
194                 const QString message = QString().sprintf("Running on an unknown WindowsNT-based system (v%u.%u).", osVersion.versionMajor, osVersion.versionMinor);
195                 qWarning("%s\n", MUTILS_UTF8(message));
196                 MUtils::OS::system_message_wrn(MUTILS_WCHR(message), L"LameXP");
197         }
198
199         //Check for compat mode
200         if(osVersion.overrideFlag && (osVersion <= MUtils::OS::Version::WINDOWS_WN100))
201         {
202                 qWarning("Windows compatibility mode detected!");
203                 if(!arguments.contains("--ignore-compat-mode", Qt::CaseInsensitive))
204                 {
205                         qFatal("%s", QApplication::tr("Executable '%1' doesn't support Windows compatibility mode.").arg(executableName).toLatin1().constData());
206                         return false;
207                 }
208         }
209
210         //Check for Wine
211         if(MUtils::OS::running_on_wine())
212         {
213                 qWarning("It appears we are running under Wine, unexpected things might happen!\n");
214         }
215
216         //Set text Codec for locale
217         QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8"));
218
219         //Create Qt application instance
220         g_application.reset(new QApplication(argc, argv));
221
222         //Load plugins from application directory
223         QCoreApplication::setLibraryPaths(QStringList() << QApplication::applicationDirPath());
224         qDebug("Library Path:\n%s\n", MUTILS_UTF8(QApplication::libraryPaths().first()));
225
226         //Set application properties
227         g_application->setApplicationName(appName);
228         g_application->setOrganizationName("LoRd_MuldeR");
229         g_application->setOrganizationDomain("mulder.at.gg");
230         g_application->setEventFilter(qt_event_filter);
231
232         //Check for supported image formats
233         QList<QByteArray> supportedFormats = QImageReader::supportedImageFormats();
234         for(int i = 0; g_imageformats[i]; i++)
235         {
236                 if(!supportedFormats.contains(g_imageformats[i]))
237                 {
238                         qFatal("Qt initialization error: QImageIOHandler for '%s' missing!", g_imageformats[i]);
239                         return false;
240                 }
241         }
242         
243         //Enable larger/smaller font size
244         double fontScaleFactor = 1.0;
245         if(arguments.contains("--huge-font",  Qt::CaseInsensitive)) fontScaleFactor = 1.500;
246         if(arguments.contains("--big-font",   Qt::CaseInsensitive)) fontScaleFactor = 1.250;
247         if(arguments.contains("--small-font", Qt::CaseInsensitive)) fontScaleFactor = 0.875;
248         if(arguments.contains("--tiny-font",  Qt::CaseInsensitive)) fontScaleFactor = 0.750;
249         if(!qFuzzyCompare(fontScaleFactor, 1.0))
250         {
251                 qWarning("Application font scale factor set to: %.3f\n", fontScaleFactor);
252                 QFont appFont = g_application->font();
253                 appFont.setPointSizeF(appFont.pointSizeF() * fontScaleFactor);
254                 g_application->setFont(appFont);
255         }
256
257         //Check for process elevation
258         if(MUtils::OS::is_elevated() && (!MUtils::OS::running_on_wine()))
259         {
260                 QMessageBox messageBox(QMessageBox::Warning, "LameXP", "<nobr>LameXP was started with 'elevated' rights, altough LameXP does not need these rights.<br>Running an applications with unnecessary rights is a potential security risk!</nobr>", QMessageBox::NoButton, NULL, Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint | Qt::WindowStaysOnTopHint);
261                 messageBox.addButton("Quit Program (Recommended)", QMessageBox::NoRole);
262                 messageBox.addButton("Ignore", QMessageBox::NoRole);
263                 if(messageBox.exec() == 0)
264                 {
265                         return NULL;
266                 }
267         }
268
269         //Successful
270         return g_application.data();
271 }
272
273 ///////////////////////////////////////////////////////////////////////////////