QGCCore.cc 9.1 KB
Newer Older
pixhawk's avatar
pixhawk committed
1 2
/*=====================================================================

pixhawk's avatar
pixhawk committed
3
QGroundControl Open Source Ground Control Station
pixhawk's avatar
pixhawk committed
4

pixhawk's avatar
pixhawk committed
5
(c) 2009, 2010 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
pixhawk's avatar
pixhawk committed
6

pixhawk's avatar
pixhawk committed
7
This file is part of the QGROUNDCONTROL project
pixhawk's avatar
pixhawk committed
8

pixhawk's avatar
pixhawk committed
9
    QGROUNDCONTROL is free software: you can redistribute it and/or modify
pixhawk's avatar
pixhawk committed
10 11 12 13
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

pixhawk's avatar
pixhawk committed
14
    QGROUNDCONTROL is distributed in the hope that it will be useful,
pixhawk's avatar
pixhawk committed
15 16 17 18 19
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
pixhawk's avatar
pixhawk committed
20
    along with QGROUNDCONTROL. If not, see <http://www.gnu.org/licenses/>.
pixhawk's avatar
pixhawk committed
21 22 23 24 25

======================================================================*/

/**
 * @file
lm's avatar
lm committed
26
 *   @brief Implementation of class QGCCore
pixhawk's avatar
pixhawk committed
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
 *
 *   @author Lorenz Meier <mavteam@student.ethz.ch>
 *
 */

#include <QFile>
#include <QFlags>
#include <QThread>
#include <QSplashScreen>
#include <QPixmap>
#include <QDesktopWidget>
#include <QPainter>
#include <QStyleFactory>
#include <QAction>

42 43
#include <QDebug>

lm's avatar
lm committed
44
#include "configuration.h"
45
#include "QGC.h"
lm's avatar
lm committed
46
#include "QGCCore.h"
pixhawk's avatar
pixhawk committed
47
#include "MainWindow.h"
pixhawk's avatar
pixhawk committed
48
#include "GAudioOutput.h"
49
#include "CmdLineOptParser.h"
pixhawk's avatar
pixhawk committed
50

51
#ifdef QGC_RTLAB_ENABLED
52 53
#include "OpalLink.h"
#endif
pixhawk's avatar
pixhawk committed
54 55
#include "UDPLink.h"
#include "MAVLinkSimulationLink.h"
56
#include "SerialLink.h"
pixhawk's avatar
pixhawk committed
57

58
const char* QGCCore::_settingsVersionKey = "SettingsVersion";
pixhawk's avatar
pixhawk committed
59 60 61 62 63 64 65 66 67 68 69

/**
 * @brief Constructor for the main application.
 *
 * This constructor initializes and starts the whole application. It takes standard
 * command-line parameters
 *
 * @param argc The number of command-line parameters
 * @param argv The string array of parameters
 **/

70

71 72 73
QGCCore::QGCCore(int &argc, char* argv[]) :
    QApplication(argc, argv),
    _mainWindow(NULL)
pixhawk's avatar
pixhawk committed
74
{
75
    // Set application name
lm's avatar
lm committed
76 77
    this->setApplicationName(QGC_APPLICATION_NAME);
    this->setApplicationVersion(QGC_APPLICATION_VERSION);
78 79
    this->setOrganizationName(QGC::ORG_NAME);
    this->setOrganizationDomain(QGC::ORG_DOMAIN);
80
    
81 82
    // Set settings format
    QSettings::setDefaultFormat(QSettings::IniFormat);
83 84 85 86 87 88 89 90 91
    
    // Parse command line options
    
    bool fClearSettingsOptions = false; // Clear stored settings
    
    CmdLineOpt_t rgCmdLineOptions[] = {
        { "--clear-settings",   &fClearSettingsOptions },
        // Add additional command line option flags here
    };
92
    
93 94
    ParseCmdLineOptions(argc, argv, rgCmdLineOptions, sizeof(rgCmdLineOptions)/sizeof(rgCmdLineOptions[0]), false);
    
95
    QSettings settings;
96
    
97 98 99
    if (fClearSettingsOptions) {
        // User requested settings to be cleared on command line
        settings.clear();
100
        settings.sync();
101 102
    }
    
103 104 105 106 107 108 109 110 111
}

bool QGCCore::init(void)
{
    QSettings settings;
    
    // Exit main application when last window is closed
    connect(this, SIGNAL(lastWindowClosed()), this, SLOT(quit()));
    
112 113
    // Show user an upgrade message if the settings version has been bumped up
    bool settingsUpgraded = false;
114
    enum MainWindow::CUSTOM_MODE mode = MainWindow::CUSTOM_MODE_PX4;
115 116 117
    if (settings.contains(_settingsVersionKey)) {
        if (settings.value(_settingsVersionKey).toInt() != QGC_SETTINGS_VERSION) {
            settingsUpgraded = true;
118
        }
119 120 121 122 123 124
    } else if (settings.allKeys().count()) {
        // Settings version key is missing and there are settings. This is an upgrade scenario.
        settingsUpgraded = true;
    }
    
    if (settingsUpgraded) {
125
        settings.clear();
126
        settings.setValue(_settingsVersionKey, QGC_SETTINGS_VERSION);
127
    }
128
    
129 130
    mode = (enum MainWindow::CUSTOM_MODE) settings.value("QGC_CUSTOM_MODE", (int)MainWindow::CUSTOM_MODE_PX4).toInt();
    
131
    settings.sync();
132
    
133
    // Show splash screen
134
    QPixmap splashImage(":/files/images/splash.png");
135
    QSplashScreen* splashScreen = new QSplashScreen(splashImage);
136 137
    // Delete splash screen after mainWindow was displayed
    splashScreen->setAttribute(Qt::WA_DeleteOnClose);
138
    splashScreen->show();
lm's avatar
lm committed
139
    processEvents();
140
    splashScreen->showMessage(tr("Loading application fonts"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
141
    
142
    // Load application font
pixhawk's avatar
pixhawk committed
143 144
    QFontDatabase fontDatabase = QFontDatabase();
    const QString fontFileName = ":/general/vera.ttf"; ///< Font file is part of the QRC file and compiled into the app
Lorenz Meier's avatar
Lorenz Meier committed
145
    //const QString fontFamilyName = "Bitstream Vera Sans";
Hugo Vincent's avatar
Hugo Vincent committed
146
    if(!QFile::exists(fontFileName)) printf("ERROR! font file: %s DOES NOT EXIST!\n", fontFileName.toStdString().c_str());
pixhawk's avatar
pixhawk committed
147
    fontDatabase.addApplicationFont(fontFileName);
148 149 150
    // Avoid Using setFont(). In the Qt docu you can read the following:
    //     "Warning: Do not use this function in conjunction with Qt Style Sheets."
    // setFont(fontDatabase.font(fontFamilyName, "Roman", 12));
151
    
pixhawk's avatar
pixhawk committed
152
    // Start the comm link manager
153
    splashScreen->showMessage(tr("Starting communication links"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
pixhawk's avatar
pixhawk committed
154
    startLinkManager();
155
    
pixhawk's avatar
pixhawk committed
156
    // Start the UAS Manager
157
    splashScreen->showMessage(tr("Starting UAS manager"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
pixhawk's avatar
pixhawk committed
158
    startUASManager();
159
    
pixhawk's avatar
pixhawk committed
160
    // Start the user interface
161
    splashScreen->showMessage(tr("Starting user interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
162
    
163
    // The first call to instance() creates the MainWindow, so make sure it's passed the splashScreen.
164 165
    _mainWindow = MainWindow::instance_mode(splashScreen, mode);
    
166
    UDPLink* udpLink = NULL;
167 168
    
    if (_mainWindow->getCustomMode() == MainWindow::CUSTOM_MODE_WIFI)
169 170 171 172 173
    {
        // Connect links
        // to make sure that all components are initialized when the
        // first messages arrive
        udpLink = new UDPLink(QHostAddress::Any, 14550);
Lorenz Meier's avatar
Lorenz Meier committed
174 175 176 177
        LinkManager::instance()->add(udpLink);
    } else {
        // We want to have a default serial link available for "quick" connecting.
        SerialLink *slink = new SerialLink();
178
        LinkManager::instance()->add(slink);
179
    }
180
    
181
#ifdef QGC_RTLAB_ENABLED
182 183
    // Add OpalRT Link, but do not connect
    OpalLink* opalLink = new OpalLink();
Lorenz Meier's avatar
Lorenz Meier committed
184
    MainWindow::instance()->addLink(opalLink);
185
#endif
186
    
lm's avatar
lm committed
187
    // Remove splash screen
188 189
    splashScreen->finish(_mainWindow);
    
190 191 192 193 194
    if (settingsUpgraded) {
        _mainWindow->showInfoMessage(tr("Settings Cleared"),
                                     tr("The format for QGroundControl saved settings has been modified. "
                                        "Your saved settings have been reset to defaults."));
    }
195
    
196
    // Check if link could be connected
197
    if (udpLink && !udpLink->connect())
198 199 200 201
    {
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Critical);
        msgBox.setText("Could not connect UDP port. Is an instance of " + qAppName() + "already running?");
202
        msgBox.setInformativeText("It is recommended to close the application and stop all instances. Click Yes to close.");
203 204 205
        msgBox.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
        msgBox.setDefaultButton(QMessageBox::No);
        int ret = msgBox.exec();
206
        
207 208
        // Close the message box shortly after the click to prevent accidental clicks
        QTimer::singleShot(15000, &msgBox, SLOT(reject()));
209
        
210 211 212 213
        // Exit application
        if (ret == QMessageBox::Yes)
        {
            //mainWindow->close();
214
            QTimer::singleShot(200, _mainWindow, SLOT(close()));
215 216
        }
    }
217 218
    
    return true;
pixhawk's avatar
pixhawk committed
219 220 221 222 223 224
}

/**
 * @brief Destructor for the groundstation. It destroys all loaded instances.
 *
 **/
lm's avatar
lm committed
225
QGCCore::~QGCCore()
pixhawk's avatar
pixhawk committed
226
{
227 228
    delete UASManager::instance();
    delete LinkManager::instance();
pixhawk's avatar
pixhawk committed
229 230 231 232 233 234 235 236
}

/**
 * @brief Start the link managing component.
 *
 * The link manager keeps track of all communication links and provides the global
 * packet queue. It is the main communication hub
 **/
lm's avatar
lm committed
237
void QGCCore::startLinkManager()
pixhawk's avatar
pixhawk committed
238 239 240 241 242 243 244 245
{
    LinkManager::instance();
}

/**
 * @brief Start the Unmanned Air System Manager
 *
 **/
lm's avatar
lm committed
246
void QGCCore::startUASManager()
pixhawk's avatar
pixhawk committed
247
{
248 249 250 251 252 253 254 255 256 257
    // Load UAS plugins
    QDir pluginsDir = QDir(qApp->applicationDirPath());

#if defined(Q_OS_WIN)
    if (pluginsDir.dirName().toLower() == "debug" || pluginsDir.dirName().toLower() == "release")
        pluginsDir.cdUp();
#elif defined(Q_OS_LINUX)
    if (pluginsDir.dirName().toLower() == "debug" || pluginsDir.dirName().toLower() == "release")
        pluginsDir.cdUp();
#elif defined(Q_OS_MAC)
258
    if (pluginsDir.dirName() == "MacOS") {
259 260 261 262 263 264 265
        pluginsDir.cdUp();
        pluginsDir.cdUp();
        pluginsDir.cdUp();
    }
#endif
    pluginsDir.cd("plugins");

lm's avatar
lm committed
266
    UASManager::instance();
267 268 269 270 271

    // Load plugins

    QStringList pluginFileNames;

272
    foreach (QString fileName, pluginsDir.entryList(QDir::Files)) {
273 274
        QPluginLoader loader(pluginsDir.absoluteFilePath(fileName));
        QObject *plugin = loader.instance();
275
        if (plugin) {
276 277
            //populateMenus(plugin);
            pluginFileNames += fileName;
lm's avatar
lm committed
278
            //printf(QString("Loaded plugin from " + fileName + "\n").toStdString().c_str());
279 280
        }
    }
pixhawk's avatar
pixhawk committed
281
}