MainWindow.cc 34.8 KB
Newer Older
1 2 3 4
/*=====================================================================

QGroundControl Open Source Ground Control Station

5
(c) 2009 - 2013 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
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

This file is part of the QGROUNDCONTROL project

    QGROUNDCONTROL is free software: you can redistribute it and/or modify
    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.

    QGROUNDCONTROL is distributed in the hope that it will be useful,
    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
    along with QGROUNDCONTROL. If not, see <http://www.gnu.org/licenses/>.

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

/**
 * @file
 *   @brief Implementation of class MainWindow
 *   @author Lorenz Meier <mail@qgroundcontrol.org>
 */

#include <QSettings>
#include <QNetworkInterface>
#include <QDebug>
#include <QTimer>
#include <QHostInfo>
#include <QSplashScreen>
dogmaphobic's avatar
dogmaphobic committed
36
#ifndef __mobile__
37 38 39
#include <QGCHilLink.h>
#include <QGCHilConfiguration.h>
#include <QGCHilFlightGearConfiguration.h>
dogmaphobic's avatar
dogmaphobic committed
40
#endif
41
#include <QQuickView>
42
#include <QDesktopWidget>
43 44
#include <QScreen>
#include <QDesktopServices>
45

46
#include "QGC.h"
dogmaphobic's avatar
dogmaphobic committed
47
#ifndef __ios__
48
#include "SerialLink.h"
dogmaphobic's avatar
dogmaphobic committed
49
#endif
50 51 52 53
#include "MAVLinkProtocol.h"
#include "MainWindow.h"
#include "GAudioOutput.h"
#include "QGCMAVLinkLogPlayer.h"
Don Gagne's avatar
Don Gagne committed
54
#include "SettingsDialog.h"
55 56
#include "MAVLinkDecoder.h"
#include "QGCMAVLinkMessageSender.h"
57
#include "UASQuickView.h"
58 59
#include "QGCDataPlot2D.h"
#include "Linecharts.h"
60 61
#include "QGCTabbedInfoView.h"
#include "UASRawStatusView.h"
62 63
#include "FlightDisplayView.h"
#include "FlightDisplayWidget.h"
64
#include "SetupView.h"
65
#include "QGCUASFileViewMulti.h"
Don Gagne's avatar
Don Gagne committed
66
#include "QGCApplication.h"
67
#include "QGCFileDialog.h"
Don Gagne's avatar
Don Gagne committed
68
#include "QGCMessageBox.h"
69
#include "QGCDockWidget.h"
70
#include "MultiVehicleManager.h"
Don Gagne's avatar
Don Gagne committed
71
#include "CustomCommandWidget.h"
72
#include "HomePositionManager.h"
Don Gagne's avatar
Don Gagne committed
73
#include "MissionEditor.h"
74

75 76 77 78
#ifdef UNITTEST_BUILD
#include "QmlControls/QmlTestWidget.h"
#endif

79 80 81 82 83 84
#ifdef QGC_OSG_ENABLED
#include "Q3DWidgetFactory.h"
#endif

#include "LogCompressor.h"

85 86 87
/// The key under which the Main Window settings are saved
const char* MAIN_SETTINGS_GROUP = "QGC_MAINWINDOW";

88
const char* MainWindow::_mavlinkDockWidgetName = "MAVLINK_INSPECTOR_DOCKWIDGET";
Don Gagne's avatar
Don Gagne committed
89
const char* MainWindow::_customCommandWidgetName = "CUSTOM_COMMAND_DOCKWIDGET";
90 91 92 93 94 95
const char* MainWindow::_filesDockWidgetName = "FILE_VIEW_DOCKWIDGET";
const char* MainWindow::_uasStatusDetailsDockWidgetName = "UAS_STATUS_DETAILS_DOCKWIDGET";
const char* MainWindow::_mapViewDockWidgetName = "MAP_VIEW_DOCKWIDGET";
const char* MainWindow::_pfdDockWidgetName = "PRIMARY_FLIGHT_DISPLAY_DOCKWIDGET";
const char* MainWindow::_uasInfoViewDockWidgetName = "UAS_INFO_INFOVIEW_DOCKWIDGET";

Don Gagne's avatar
Don Gagne committed
96 97
static MainWindow* _instance = NULL;   ///< @brief MainWindow singleton

98
MainWindow* MainWindow::_create(QSplashScreen* splashScreen)
99
{
Don Gagne's avatar
Don Gagne committed
100
    Q_ASSERT(_instance == NULL);
101
    new MainWindow(splashScreen);
Don Gagne's avatar
Don Gagne committed
102 103
    // _instance is set in constructor
    Q_ASSERT(_instance);
104 105 106
    return _instance;
}

Don Gagne's avatar
Don Gagne committed
107
MainWindow* MainWindow::instance(void)
108
{
Don Gagne's avatar
Don Gagne committed
109
    return _instance;
110 111
}

112 113
void MainWindow::deleteInstance(void)
{
Don Gagne's avatar
Don Gagne committed
114
    delete this;
115 116
}

Don Gagne's avatar
Don Gagne committed
117 118 119
/// @brief Private constructor for MainWindow. MainWindow singleton is only ever created
///         by MainWindow::_create method. Hence no other code should have access to
///         constructor.
dogmaphobic's avatar
dogmaphobic committed
120 121 122
MainWindow::MainWindow(QSplashScreen* splashScreen)
    : _autoReconnect(false)
    , _lowPowerMode(false)
123
    , _showStatusBar(false)
dogmaphobic's avatar
dogmaphobic committed
124 125 126 127 128
    , _centerStackActionGroup(new QActionGroup(this))
    , _centralLayout(NULL)
    , _currentViewWidget(NULL)
    , _splashScreen(splashScreen)
    , _currentView(VIEW_SETUP)
129
{
Don Gagne's avatar
Don Gagne committed
130 131
    Q_ASSERT(_instance == NULL);
    _instance = this;
132

133 134 135
    if (splashScreen) {
        connect(this, &MainWindow::initStatusChanged, splashScreen, &QSplashScreen::showMessage);
    }
136

137 138 139 140
    // Qt 4/5 on Ubuntu does place the native menubar correctly so on Linux we revert back to in-window menu bar.
#ifdef Q_OS_LINUX
    menuBar()->setNativeMenuBar(false);
#endif
dogmaphobic's avatar
dogmaphobic committed
141
    // Setup user interface
142
    loadSettings();
143
    emit initStatusChanged(tr("Setting up user interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
dogmaphobic's avatar
dogmaphobic committed
144 145
    _ui.setupUi(this);
    // Make sure tool bar elements all fit before changing minimum width
dogmaphobic's avatar
dogmaphobic committed
146
    setMinimumWidth(1008);
dogmaphobic's avatar
dogmaphobic committed
147
    configureWindowName();
148

149 150
    // Setup central widget with a layout to hold the views
    _centralLayout = new QVBoxLayout();
151
    _centralLayout->setContentsMargins(0,0,0,0);
152
    centralWidget()->setLayout(_centralLayout);
153 154 155
    // Set dock options
    setDockOptions(AnimatedDocks | AllowTabbedDocks | AllowNestedDocks);
    // Setup corners
156
    setCorner(Qt::BottomRightCorner, Qt::BottomDockWidgetArea);
157

dogmaphobic's avatar
dogmaphobic committed
158 159 160
    // On Mobile devices, we don't want any main menus at all.
#ifdef __mobile__
    menuBar()->setNativeMenuBar(false);
dogmaphobic's avatar
dogmaphobic committed
161
#endif
dogmaphobic's avatar
dogmaphobic committed
162

163 164 165
#ifdef UNITTEST_BUILD
    QAction* qmlTestAction = new QAction("Test QML palette and controls", NULL);
    connect(qmlTestAction, &QAction::triggered, this, &MainWindow::_showQmlTestWidget);
dogmaphobic's avatar
dogmaphobic committed
166
    _ui.menuTools->addAction(qmlTestAction);
167
#endif
168

dogmaphobic's avatar
dogmaphobic committed
169 170 171 172
    // Load QML Toolbar
    QDockWidget* widget = new QDockWidget(this);
    widget->setObjectName("ToolBarDockWidget");
    qmlRegisterType<MainToolBar>("QGroundControl.MainToolBar", 1, 0, "MainToolBar");
173
    _mainToolBar = new MainToolBar(widget);
dogmaphobic's avatar
dogmaphobic committed
174 175 176 177
    widget->setWidget(_mainToolBar);
    widget->setFeatures(QDockWidget::NoDockWidgetFeatures);
    widget->setTitleBarWidget(new QWidget(this)); // Disables the title bar
    addDockWidget(Qt::TopDockWidgetArea, widget);
178

dogmaphobic's avatar
dogmaphobic committed
179 180 181
    // Setup UI state machines
    _centerStackActionGroup->setExclusive(true);
    // Status Bar
182
    setStatusBar(new QStatusBar(this));
183
    statusBar()->setSizeGripEnabled(true);
184

185
    emit initStatusChanged(tr("Building common widgets."), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
186
    _buildCommonWidgets();
187
    emit initStatusChanged(tr("Building common actions"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
188 189 190
    // Create actions
    connectCommonActions();
    // Connect user interface devices
191
#ifdef QGC_MOUSE_ENABLED_WIN
192
    emit initStatusChanged(tr("Initializing 3D mouse interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
193 194
    mouseInput = new Mouse3DInput(this);
    mouse = new Mouse6dofInput(mouseInput);
195
#endif //QGC_MOUSE_ENABLED_WIN
196

197
#if QGC_MOUSE_ENABLED_LINUX
198
    emit initStatusChanged(tr("Initializing 3D mouse interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
199 200

    mouse = new Mouse6dofInput(this);
201
    connect(this, SIGNAL(x11EventOccured(XEvent*)), mouse, SLOT(handleX11Event(XEvent*)));
202
#endif //QGC_MOUSE_ENABLED_LINUX
203

204 205 206 207
    // These also cause the screen to redraw so we need to update any OpenGL canvases in QML controls
    connect(LinkManager::instance(), &LinkManager::linkConnected,    this, &MainWindow::_linkStateChange);
    connect(LinkManager::instance(), &LinkManager::linkDisconnected, this, &MainWindow::_linkStateChange);

208
    // Connect link
dogmaphobic's avatar
dogmaphobic committed
209
    if (_autoReconnect)
210
    {
211
        restoreLastUsedConnection();
212 213 214
    }

    // Set low power mode
dogmaphobic's avatar
dogmaphobic committed
215
    enableLowPowerMode(_lowPowerMode);
216
    emit initStatusChanged(tr("Restoring last view state"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
217
    // Restore the window setup
218
    _loadCurrentViewState();
dogmaphobic's avatar
dogmaphobic committed
219
#ifndef __mobile__
220

221
    // Restore the window position and size
dogmaphobic's avatar
dogmaphobic committed
222 223
    emit initStatusChanged(tr("Restoring last window size"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    if (settings.contains(_getWindowGeometryKey()))
224
    {
dogmaphobic's avatar
dogmaphobic committed
225
        restoreGeometry(settings.value(_getWindowGeometryKey()).toByteArray());
226 227 228 229
    }
    else
    {
        // Adjust the size
230 231 232
        QScreen* scr = QApplication::primaryScreen();
        QSize scrSize = scr->availableSize();
        if (scrSize.width() <= 1280)
233
        {
234
            resize(scrSize.width(), scrSize.height());
235 236 237
        }
        else
        {
238 239 240
            int w = scrSize.width()  > 1600 ? 1600 : scrSize.width();
            int h = scrSize.height() >  800 ?  800 : scrSize.height();
            resize(w, h);
241
            move((scrSize.width() - w) / 2, (scrSize.height() - h) / 2);
242 243 244
        }
    }

245 246 247
    // Make sure the proper fullscreen/normal menu item is checked properly.
    if (isFullScreen())
    {
dogmaphobic's avatar
dogmaphobic committed
248 249
        _ui.actionFullscreen->setChecked(true);
        _ui.actionNormal->setChecked(false);
250 251 252
    }
    else
    {
dogmaphobic's avatar
dogmaphobic committed
253 254
        _ui.actionFullscreen->setChecked(false);
        _ui.actionNormal->setChecked(true);
255 256 257
    }

    // And that they will stay checked properly after user input
258 259
    connect(_ui.actionFullscreen, &QAction::triggered, this, &MainWindow::fullScreenActionItemCallback);
    connect(_ui.actionNormal,     &QAction::triggered, this, &MainWindow::normalActionItemCallback);
260 261
#endif

262
    connect(_ui.actionStatusBar,  &QAction::triggered, this, &MainWindow::showStatusBarCallback);
263

264 265
    // Set OS dependent keyboard shortcuts for the main window, non OS dependent shortcuts are set in MainWindow.ui
#ifdef Q_OS_MACX
dogmaphobic's avatar
dogmaphobic committed
266
    _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Meta+1", 0));
267 268 269
    _ui.actionPlan->setShortcut(QApplication::translate("MainWindow", "Meta+2", 0));
    _ui.actionFlight->setShortcut(QApplication::translate("MainWindow", "Meta+3", 0));
    _ui.actionAnalyze->setShortcut(QApplication::translate("MainWindow", "Meta+4", 0));
Don Gagne's avatar
Don Gagne committed
270
    _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Meta+5", 0));
dogmaphobic's avatar
dogmaphobic committed
271
    _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Meta+Return", 0));
272
#else
dogmaphobic's avatar
dogmaphobic committed
273
    _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Ctrl+1", 0));
Gus Grubba's avatar
Gus Grubba committed
274
    _ui.actionPlan->setShortcut(QApplication::translate("MainWindow", "Ctrl+2", 0));
275
    _ui.actionFlight->setShortcut(QApplication::translate("MainWindow", "Ctrl+3", 0));
Gus Grubba's avatar
Gus Grubba committed
276
    _ui.actionAnalyze->setShortcut(QApplication::translate("MainWindow", "Ctrl+4", 0));
Don Gagne's avatar
Don Gagne committed
277
    _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Ctrl+5", 0));
dogmaphobic's avatar
dogmaphobic committed
278
    _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Ctrl+Return", 0));
279 280
#endif

281 282
    connect(&windowNameUpdateTimer, SIGNAL(timeout()), this, SLOT(configureWindowName()));
    windowNameUpdateTimer.start(15000);
283
    emit initStatusChanged(tr("Done"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
284 285

    if (!qgcApp()->runningUnitTests()) {
286 287
        _ui.actionStatusBar->setChecked(_showStatusBar);
        showStatusBarCallback(_showStatusBar);
dogmaphobic's avatar
dogmaphobic committed
288
#ifdef __mobile__
289 290
        menuBar()->hide();
#endif
291
        show();
dogmaphobic's avatar
dogmaphobic committed
292 293 294 295 296 297 298 299 300 301 302 303 304 305 306
#ifdef Q_OS_MAC
        // TODO HACK
        // This is a really ugly hack. For whatever reason, by having a QQuickWidget inside a
        // QDockWidget (MainToolBar above), the main menu is not shown when the app first
        // starts. I looked everywhere and I could not find a solution. What I did notice was
        // that if any other window gets focus, the menu comes up when you come back to QGC.
        // That is, if you were to click on another window and then back to QGC, the menus
        // would appear. This hack below creates a 0x0 dialog and immediately closes it.
        // That works around the issue and it will do until I find the root of the problem.
        QDialog qd(this);
        qd.show();
        qd.raise();
        qd.activateWindow();
        qd.close();
#endif
307
    }
308 309 310 311 312
}

MainWindow::~MainWindow()
{
    // Delete all UAS objects
dogmaphobic's avatar
dogmaphobic committed
313
    for (int i=0;i<_commsWidgetList.size();i++)
314
    {
dogmaphobic's avatar
dogmaphobic committed
315
        _commsWidgetList[i]->deleteLater();
316
    }
Don Gagne's avatar
Don Gagne committed
317
    _instance = NULL;
318 319 320 321 322 323 324
}

void MainWindow::resizeEvent(QResizeEvent * event)
{
    QMainWindow::resizeEvent(event);
}

dogmaphobic's avatar
dogmaphobic committed
325
QString MainWindow::_getWindowStateKey()
326
{
327
	return QString::number(_currentView)+"_windowstate_";
328 329
}

dogmaphobic's avatar
dogmaphobic committed
330
QString MainWindow::_getWindowGeometryKey()
331 332 333 334
{
    return "_geometry";
}

335 336 337
void MainWindow::_createDockWidget(const QString& title, const QString& name, Qt::DockWidgetArea area, QWidget* innerWidget)
{
    Q_ASSERT(!_mapName2DockWidget.contains(name));
338
	
339 340 341 342 343
    // Add to menu
    QAction* action = new QAction(title, NULL);
    action->setCheckable(true);
    action->setData(name);
    connect(action, &QAction::triggered, this, &MainWindow::_showDockWidgetAction);
dogmaphobic's avatar
dogmaphobic committed
344
    _ui.menuTools->addAction(action);
345 346 347 348 349 350 351 352 353 354 355 356 357
	
	// Create widget
	QGCDockWidget* dockWidget = new QGCDockWidget(title, action, this);
	Q_CHECK_PTR(dockWidget);
	dockWidget->setObjectName(name);
	dockWidget->setVisible (false);
	if (innerWidget) {
		// Put inner widget inside QDockWidget
		innerWidget->setParent(dockWidget);
		dockWidget->setWidget(innerWidget);
		innerWidget->setVisible(true);
	}
	
358 359 360 361 362 363
    _mapName2DockWidget[name] = dockWidget;
    _mapDockWidget2Action[dockWidget] = action;
    addDockWidget(area, dockWidget);
}

void MainWindow::_buildCommonWidgets(void)
364 365
{
    // Add generic MAVLink decoder
dogmaphobic's avatar
dogmaphobic committed
366
    // TODO: This is never deleted
367
    mavlinkDecoder = new MAVLinkDecoder(MAVLinkProtocol::instance(), this);
John Tapsell's avatar
John Tapsell committed
368 369
    connect(mavlinkDecoder, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)),
                      this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
370

371
    // Log player
dogmaphobic's avatar
dogmaphobic committed
372
    // TODO: Make this optional with a preferences setting or under a "View" menu
Don Gagne's avatar
Don Gagne committed
373
    logPlayer = new QGCMAVLinkLogPlayer(statusBar());
374
    statusBar()->addPermanentWidget(logPlayer);
375

376 377 378
    // In order for Qt to save and restore state of widgets all widgets must be created ahead of time. We only create the QDockWidget
    // holders. We do not create the actual inner widget until it is needed. This saves memory and cpu from running widgets that are
    // never shown.
379

380 381 382 383 384
    struct DockWidgetInfo {
        const char* name;
        const char* title;
        Qt::DockWidgetArea area;
    };
385

386 387
    static const struct DockWidgetInfo rgDockWidgetInfo[] = {
        { _mavlinkDockWidgetName,           "MAVLink Inspector",        Qt::RightDockWidgetArea },
Don Gagne's avatar
Don Gagne committed
388
        { _customCommandWidgetName,         "Custom Command",			Qt::RightDockWidgetArea },
389 390 391 392 393 394 395
        { _filesDockWidgetName,             "Onboard Files",            Qt::RightDockWidgetArea },
        { _uasStatusDetailsDockWidgetName,  "Status Details",           Qt::RightDockWidgetArea },
        { _mapViewDockWidgetName,           "Map view",                 Qt::RightDockWidgetArea },
        { _pfdDockWidgetName,               "Primary Flight Display",   Qt::RightDockWidgetArea },
        { _uasInfoViewDockWidgetName,       "Info View",                Qt::LeftDockWidgetArea },
    };
    static const size_t cDockWidgetInfo = sizeof(rgDockWidgetInfo) / sizeof(rgDockWidgetInfo[0]);
396

397 398 399
    for (size_t i=0; i<cDockWidgetInfo; i++) {
        const struct DockWidgetInfo* pDockInfo = &rgDockWidgetInfo[i];
        _createDockWidget(pDockInfo->title, pDockInfo->name, pDockInfo->area, NULL /* no inner widget yet */);
400
    }
401
}
402

Don Gagne's avatar
Don Gagne committed
403 404 405 406 407 408 409 410
void MainWindow::_buildMissionEditorView(void)
{
    if (!_missionEditorView) {
        _missionEditorView = new MissionEditor(this);
        _missionEditorView->setVisible(false);
    }
}

411 412 413
void MainWindow::_buildFlightView(void)
{
    if (!_flightView) {
414
        _flightView = new FlightDisplayView(this);
415
        _flightView->setVisible(false);
416
    }
417 418
}

419
void MainWindow::_buildSetupView(void)
420
{
421 422 423 424
    if (!_setupView) {
        _setupView = new SetupView(this);
        _setupView->setVisible(false);
    }
425 426
}

427
void MainWindow::_buildAnalyzeView(void)
428
{
429 430 431
    if (!_analyzeView) {
        _analyzeView = new QGCDataPlot2D(this);
        _analyzeView->setVisible(false);
432 433
    }
}
434

435 436 437
void MainWindow::_buildSimView(void)
{
    if (!_simView) {
438
        _simView = new FlightDisplayView(this);
439 440
        _simView->setVisible(false);
    }
441
}
442

443 444
/// Shows or hides the specified dock widget, creating if necessary
void MainWindow::_showDockWidget(const QString& name, bool show)
445
{
446
    if (!_mapName2DockWidget.contains(name)) {
Don Gagne's avatar
Don Gagne committed
447 448
        // Don't show any sort of warning here. Dock Widgets which have been remove could still be in settings.
        // Which would cause us to end up here.
449 450
        return;
    }
451

452 453 454
    // Create the inner widget if we need to
    if (!_mapName2DockWidget[name]->widget()) {
        _createInnerDockWidget(name);
455
    }
456 457 458 459

    Q_ASSERT(_mapName2DockWidget.contains(name));
    QDockWidget* dockWidget = _mapName2DockWidget[name];
    Q_ASSERT(dockWidget);
460

461
    dockWidget->setVisible(show);
462

463 464 465 466 467 468 469 470 471
    Q_ASSERT(_mapDockWidget2Action.contains(dockWidget));
    _mapDockWidget2Action[dockWidget]->setChecked(show);
}

/// Creates the specified inner dock widget and adds to the QDockWidget
void MainWindow::_createInnerDockWidget(const QString& widgetName)
{
    Q_ASSERT(_mapName2DockWidget.contains(widgetName)); // QDockWidget should already exist
    Q_ASSERT(!_mapName2DockWidget[widgetName]->widget());     // Inner widget should not
472

473
    QWidget* widget = NULL;
474

475
    if (widgetName == _mavlinkDockWidgetName) {
476
        widget = new QGCMAVLinkInspector(MAVLinkProtocol::instance(),this);
Don Gagne's avatar
Don Gagne committed
477 478
    } else if (widgetName == _customCommandWidgetName) {
        widget = new CustomCommandWidget(this);
479 480 481 482 483
    } else if (widgetName == _filesDockWidgetName) {
        widget = new QGCUASFileViewMulti(this);
    } else if (widgetName == _uasStatusDetailsDockWidgetName) {
        widget = new UASInfoWidget(this);
    } else if (widgetName == _pfdDockWidgetName) {
484
        widget = new FlightDisplayWidget(this);
485
    } else if (widgetName == _uasInfoViewDockWidgetName) {
486 487 488
        QGCTabbedInfoView* pInfoView = new QGCTabbedInfoView(this);
        pInfoView->addSource(mavlinkDecoder);
        widget = pInfoView;
489 490
    } else {
        qWarning() << "Attempt to create unknown Inner Dock Widget" << widgetName;
491
    }
492

493 494 495 496 497
    if (widget) {
        QDockWidget* dockWidget = _mapName2DockWidget[widgetName];
        Q_CHECK_PTR(dockWidget);
        widget->setParent(dockWidget);
        dockWidget->setWidget(widget);
498 499
    }
}
500

dogmaphobic's avatar
dogmaphobic committed
501
#ifndef __mobile__
502
void MainWindow::_showHILConfigurationWidgets(void)
503
{
504
    Vehicle* vehicle = MultiVehicleManager::instance()->activeVehicle();
505

506
    if (!vehicle) {
507 508
        return;
    }
509

510
    UAS* mav = vehicle->uas();
511
    Q_ASSERT(mav);
512

513
    int uasId = mav->getUASID();
514

515
    if (!_mapUasId2HilDockWidget.contains(uasId)) {
516

517
        // Create QDockWidget
518
        QGCDockWidget* dockWidget = new QGCDockWidget(tr("HIL Config %1").arg(uasId), NULL, this);
519 520 521
        Q_CHECK_PTR(dockWidget);
        dockWidget->setObjectName(tr("HIL_CONFIG_%1").arg(uasId));
        dockWidget->setVisible (false);
522

523 524
        // Create inner widget and set it
        QWidget* widget = new QGCHilConfiguration(mav, dockWidget);
525

526 527
        widget->setParent(dockWidget);
        dockWidget->setWidget(widget);
528

529
        _mapUasId2HilDockWidget[uasId] = dockWidget;
530

531 532
        addDockWidget(Qt::LeftDockWidgetArea, dockWidget);
    }
533

534 535 536 537 538
    if (_currentView == VIEW_SIMULATION) {
        // HIL dock widgets only show up on simulation view
        foreach (QDockWidget* dockWidget, _mapUasId2HilDockWidget) {
            dockWidget->setVisible(true);
        }
539 540
    }
}
dogmaphobic's avatar
dogmaphobic committed
541
#endif
542

543
void MainWindow::fullScreenActionItemCallback(bool)
544
{
dogmaphobic's avatar
dogmaphobic committed
545
    _ui.actionNormal->setChecked(false);
546 547
}

548
void MainWindow::normalActionItemCallback(bool)
549
{
dogmaphobic's avatar
dogmaphobic committed
550
    _ui.actionFullscreen->setChecked(false);
551 552
}

553 554 555 556 557 558
void MainWindow::showStatusBarCallback(bool checked)
{
    _showStatusBar = checked;
    checked ? statusBar()->show() : statusBar()->hide();
}

559 560
void MainWindow::closeEvent(QCloseEvent *event)
{
561
    // Disallow window close if there are active connections
562
    if (LinkManager::instance()->anyConnectedLinks()) {
dogmaphobic's avatar
dogmaphobic committed
563 564 565 566 567 568
        QGCMessageBox::StandardButton button =
            QGCMessageBox::warning(
                tr("QGroundControl close"),
                tr("There are still active connections to vehicles. Do you want to disconnect these before closing?"),
                QMessageBox::Yes | QMessageBox::Cancel,
                QMessageBox::Cancel);
569 570 571 572 573 574 575 576 577 578
		if (button == QMessageBox::Yes) {
			LinkManager::instance()->disconnectAll();
			// The above disconnect causes a flurry of activity as the vehicle components are removed. This in turn
			// causes the Windows Version of Qt to crash if you allow the close event to be accepted. In order to prevent
			// the crash, we ignore the close event and setup a delayed timer to close the window after things settle down.
			QTimer::singleShot(1500, this, &MainWindow::_closeWindow);
		}

        event->ignore();
        return;
579 580
    }

581 582
    // This will process any remaining flight log save dialogs
    qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents);
583
    
584
    // Should not be any active connections
585 586
    Q_ASSERT(!LinkManager::instance()->anyConnectedLinks());
    
587
    _storeCurrentViewState();
588
    storeSettings();
589
    event->accept();
590 591 592 593
}

void MainWindow::loadSettings()
{
594
    // Why the screaming?
595
    QSettings settings;
596
    settings.beginGroup(MAIN_SETTINGS_GROUP);
597 598 599
    _autoReconnect  = settings.value("AUTO_RECONNECT",      _autoReconnect).toBool();
    _lowPowerMode   = settings.value("LOW_POWER_MODE",      _lowPowerMode).toBool();
    _showStatusBar  = settings.value("SHOW_STATUSBAR",      _showStatusBar).toBool();
600 601 602 603 604 605
    settings.endGroup();
}

void MainWindow::storeSettings()
{
    QSettings settings;
606
    settings.beginGroup(MAIN_SETTINGS_GROUP);
607 608 609
    settings.setValue("AUTO_RECONNECT",     _autoReconnect);
    settings.setValue("LOW_POWER_MODE",     _lowPowerMode);
    settings.setValue("SHOW_STATUSBAR",     _showStatusBar);
610
    settings.endGroup();
dogmaphobic's avatar
dogmaphobic committed
611
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
612
	
613
    // Save the last current view in any case
614
    settings.setValue("CURRENT_VIEW", _currentView);
615
    settings.setValue(_getWindowStateKey(), saveState());
616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639
}

void MainWindow::configureWindowName()
{
    QList<QHostAddress> hostAddresses = QNetworkInterface::allAddresses();
    QString windowname = qApp->applicationName() + " " + qApp->applicationVersion();
    bool prevAddr = false;
    windowname.append(" (" + QHostInfo::localHostName() + ": ");
    for (int i = 0; i < hostAddresses.size(); i++)
    {
        // Exclude loopback IPv4 and all IPv6 addresses
        if (hostAddresses.at(i) != QHostAddress("127.0.0.1") && !hostAddresses.at(i).toString().contains(":"))
        {
            if(prevAddr) windowname.append("/");
            windowname.append(hostAddresses.at(i).toString());
            prevAddr = true;
        }
    }
    windowname.append(")");
    setWindowTitle(windowname);
}

void MainWindow::enableAutoReconnect(bool enabled)
{
dogmaphobic's avatar
dogmaphobic committed
640
    _autoReconnect = enabled;
641 642 643 644 645 646 647 648 649
}

/**
* @brief Create all actions associated to the main window
*
**/
void MainWindow::connectCommonActions()
{
    // Bind together the perspective actions
dogmaphobic's avatar
dogmaphobic committed
650
    QActionGroup* perspectives = new QActionGroup(_ui.menuPerspectives);
651 652
    perspectives->addAction(_ui.actionAnalyze);
    perspectives->addAction(_ui.actionFlight);
dogmaphobic's avatar
dogmaphobic committed
653
    perspectives->addAction(_ui.actionSimulationView);
654
    perspectives->addAction(_ui.actionPlan);
dogmaphobic's avatar
dogmaphobic committed
655
    perspectives->addAction(_ui.actionSetup);
656 657 658
    perspectives->setExclusive(true);

    // Mark the right one as selected
659
    if (_currentView == VIEW_ANALYZE)
660
    {
661 662
        _ui.actionAnalyze->setChecked(true);
        _ui.actionAnalyze->activate(QAction::Trigger);
663
    }
664
    if (_currentView == VIEW_FLIGHT)
665
    {
666 667
        _ui.actionFlight->setChecked(true);
        _ui.actionFlight->activate(QAction::Trigger);
668
    }
669
    if (_currentView == VIEW_SIMULATION)
670
    {
dogmaphobic's avatar
dogmaphobic committed
671 672
        _ui.actionSimulationView->setChecked(true);
        _ui.actionSimulationView->activate(QAction::Trigger);
673
    }
674
    if (_currentView == VIEW_MISSIONEDITOR)
675
    {
676 677 678
        _ui.actionPlan->setChecked(true);
        _ui.actionPlan->activate(QAction::Trigger);
    }
679
    if (_currentView == VIEW_SETUP)
680
    {
dogmaphobic's avatar
dogmaphobic committed
681 682
        _ui.actionSetup->setChecked(true);
        _ui.actionSetup->activate(QAction::Trigger);
683
    }
684 685

    // Connect actions from ui
dogmaphobic's avatar
dogmaphobic committed
686
    connect(_ui.actionAdd_Link, SIGNAL(triggered()), this, SLOT(manageLinks()));
687 688

    // Connect internal actions
689 690
    connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleAdded, this, &MainWindow::_vehicleAdded);
    connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleRemoved, this, &MainWindow::_vehicleRemoved);
691 692

    // Views actions
Don Gagne's avatar
Don Gagne committed
693 694 695 696
    connect(_ui.actionFlight,           SIGNAL(triggered()), this, SLOT(loadFlightView()));
    connect(_ui.actionSimulationView,   SIGNAL(triggered()), this, SLOT(loadSimulationView()));
    connect(_ui.actionAnalyze,          SIGNAL(triggered()), this, SLOT(loadAnalyzeView()));
    connect(_ui.actionPlan,             SIGNAL(triggered()), this, SLOT(loadPlanView()));
697
    
698
    // Help Actions
dogmaphobic's avatar
dogmaphobic committed
699 700 701
    connect(_ui.actionOnline_Documentation, SIGNAL(triggered()), this, SLOT(showHelp()));
    connect(_ui.actionDeveloper_Credits, SIGNAL(triggered()), this, SLOT(showCredits()));
    connect(_ui.actionProject_Roadmap, SIGNAL(triggered()), this, SLOT(showRoadMap()));
702 703

    // Audio output
dogmaphobic's avatar
dogmaphobic committed
704 705 706
    _ui.actionMuteAudioOutput->setChecked(GAudioOutput::instance()->isMuted());
    connect(GAudioOutput::instance(), SIGNAL(mutedChanged(bool)), _ui.actionMuteAudioOutput, SLOT(setChecked(bool)));
    connect(_ui.actionMuteAudioOutput, SIGNAL(triggered(bool)), GAudioOutput::instance(), SLOT(mute(bool)));
707 708

    // Application Settings
dogmaphobic's avatar
dogmaphobic committed
709
    connect(_ui.actionSettings, SIGNAL(triggered()), this, SLOT(showSettings()));
710

dogmaphobic's avatar
dogmaphobic committed
711
    // Update Tool Bar
712
    _mainToolBar->setCurrentView(_currentView);
713 714
}

Don Gagne's avatar
Don Gagne committed
715
void MainWindow::_openUrl(const QString& url, const QString& errorMessage)
716
{
Don Gagne's avatar
Don Gagne committed
717
    if(!QDesktopServices::openUrl(QUrl(url))) {
dogmaphobic's avatar
dogmaphobic committed
718 719 720 721
        QMessageBox::critical(
            this,
            tr("Could not open information in browser"),
            errorMessage);
722 723 724
    }
}

Don Gagne's avatar
Don Gagne committed
725 726
void MainWindow::showHelp()
{
dogmaphobic's avatar
dogmaphobic committed
727 728 729
    _openUrl(
        "http://qgroundcontrol.org/users/start",
        tr("To get to the online help, please open http://qgroundcontrol.org/user_guide in a browser."));
Don Gagne's avatar
Don Gagne committed
730 731
}

732 733
void MainWindow::showCredits()
{
dogmaphobic's avatar
dogmaphobic committed
734 735 736
    _openUrl(
        "http://qgroundcontrol.org/credits",
        tr("To get to the credits, please open http://qgroundcontrol.org/credits in a browser."));
737 738 739 740
}

void MainWindow::showRoadMap()
{
dogmaphobic's avatar
dogmaphobic committed
741 742 743
    _openUrl(
        "http://qgroundcontrol.org/dev/roadmap",
        tr("To get to the online help, please open http://qgroundcontrol.org/roadmap in a browser."));
744 745 746 747
}

void MainWindow::showSettings()
{
dogmaphobic's avatar
dogmaphobic committed
748
    SettingsDialog settings(this);
Don Gagne's avatar
Don Gagne committed
749
    settings.exec();
750 751
}

752 753
void MainWindow::commsWidgetDestroyed(QObject *obj)
{
754 755
    // Do not dynamic cast or de-reference QObject, since object is either in destructor or may have already
    // been destroyed.
dogmaphobic's avatar
dogmaphobic committed
756
    if (_commsWidgetList.contains(obj))
757
    {
dogmaphobic's avatar
dogmaphobic committed
758
        _commsWidgetList.removeOne(obj);
759 760
    }
}
761

762
void MainWindow::_vehicleAdded(Vehicle* vehicle)
763
{
764
    connect(vehicle->uas(), SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)), this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
765

766
    // HIL
dogmaphobic's avatar
dogmaphobic committed
767
#ifndef __mobile__
768
    _showHILConfigurationWidgets();
dogmaphobic's avatar
dogmaphobic committed
769
#endif
770

771 772 773
    if (!linechartWidget)
    {
        linechartWidget = new Linecharts(this);
774
        linechartWidget->setVisible(false);
775
    }
776

777
    linechartWidget->addSource(mavlinkDecoder);
778
    if (_analyzeView != linechartWidget)
779
    {
780
        _analyzeView = linechartWidget;
781
    }
782 783
}

784
void MainWindow::_vehicleRemoved(Vehicle* vehicle)
785
{
786 787 788 789 790
    int vehicleId = vehicle->id();
    
    if (_mapUasId2HilDockWidget.contains(vehicleId)) {
        _mapUasId2HilDockWidget[vehicleId]->deleteLater();
        _mapUasId2HilDockWidget.remove(vehicleId);
791 792 793
    }
}

794 795
/// Stores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_storeCurrentViewState(void)
796
{
797 798
    // HIL dock widgets are dynamic and are not part of the saved state
    _hideAllHilDockWidgets();
Don Gagne's avatar
Don Gagne committed
799 800
    
#ifndef __mobile__
801 802 803 804 805 806 807 808 809 810 811
    // Save list of visible widgets
    bool firstWidget = true;
    QString widgetNames = "";
    foreach(QDockWidget* dockWidget, _mapName2DockWidget) {
        if (dockWidget->isVisible()) {
            if (!firstWidget) {
                widgetNames += ",";
            }
            widgetNames += dockWidget->objectName();
            firstWidget = false;
        }
812
    }
dogmaphobic's avatar
dogmaphobic committed
813
    settings.setValue(_getWindowStateKey() + "WIDGETS", widgetNames);
Don Gagne's avatar
Don Gagne committed
814
#endif
dogmaphobic's avatar
dogmaphobic committed
815 816
    settings.setValue(_getWindowStateKey(), saveState());
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
817 818
}

819 820
/// Restores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_loadCurrentViewState(void)
821
{
Don Gagne's avatar
Don Gagne committed
822
    QWidget* centerView = NULL;
823
    QString defaultWidgets;
824

825
    switch (_currentView) {
826
        case VIEW_SETUP:
827 828
            _buildSetupView();
            centerView = _setupView;
829
            break;
830

831 832 833
        case VIEW_ANALYZE:
            _buildAnalyzeView();
            centerView = _analyzeView;
834
            defaultWidgets = "PARAMETER_INTERFACE_DOCKWIDGET,FILE_VIEW_DOCKWIDGET";
835
            break;
836

837
        case VIEW_FLIGHT:
838 839
            _buildFlightView();
            centerView = _flightView;
840
            defaultWidgets = "COMMUNICATION_CONSOLE_DOCKWIDGET,UAS_INFO_INFOVIEW_DOCKWIDGET";
841
            break;
842

Don Gagne's avatar
Don Gagne committed
843 844 845 846 847
        case VIEW_MISSIONEDITOR:
            _buildMissionEditorView();
            centerView = _missionEditorView;
            break;

848
        case VIEW_SIMULATION:
849 850
            _buildSimView();
            centerView = _simView;
Don Gagne's avatar
Don Gagne committed
851
            defaultWidgets = "WAYPOINT_LIST_DOCKWIDGET,PARAMETER_INTERFACE_DOCKWIDGET,PRIMARY_FLIGHT_DISPLAY_DOCKWIDGET";
852
            break;
853

854 855
        default:
            Q_ASSERT(false);
856
            break;
857
    }
858

859 860 861 862 863 864 865 866
    // Remove old view
    if (_currentViewWidget) {
        _currentViewWidget->setVisible(false);
        Q_ASSERT(_centralLayout->count() == 1);
        QLayoutItem *child = _centralLayout->takeAt(0);
        Q_ASSERT(child);
        delete child;
    }
867

868 869 870 871 872
    // Add the new one
    Q_ASSERT(centerView);
    Q_ASSERT(_centralLayout->count() == 0);
    _currentViewWidget = centerView;
    _centralLayout->addWidget(_currentViewWidget);
873
    _centralLayout->setContentsMargins(0, 0, 0, 0);
874
    _currentViewWidget->setVisible(true);
875

876 877 878
    // Hide all widgets from previous view
    _hideAllDockWidgets();

Don Gagne's avatar
Don Gagne committed
879
#ifndef __mobile__
880
    // Restore the widgets for the new view
dogmaphobic's avatar
dogmaphobic committed
881
    QString widgetNames = settings.value(_getWindowStateKey() + "WIDGETS", defaultWidgets).toString();
882
    qDebug() << widgetNames;
883 884 885 886 887
    if (!widgetNames.isEmpty()) {
        QStringList split = widgetNames.split(",");
        foreach (QString widgetName, split) {
            Q_ASSERT(!widgetName.isEmpty());
            _showDockWidget(widgetName, true);
888 889
        }
    }
Don Gagne's avatar
Don Gagne committed
890
#endif
891

dogmaphobic's avatar
dogmaphobic committed
892 893
    if (settings.contains(_getWindowStateKey())) {
        restoreState(settings.value(_getWindowStateKey()).toByteArray());
894
    }
895

896 897
    // HIL dock widget are dynamic and don't take part in the saved window state, so this
    // need to happen after we restore state
dogmaphobic's avatar
dogmaphobic committed
898
#ifndef __mobile__
899
    _showHILConfigurationWidgets();
dogmaphobic's avatar
dogmaphobic committed
900
#endif
901 902 903 904

    // There is a bug in Qt where a Canvas element inside a QQuickWidget does not
    // receive update requests. Here we emit a signal for them to get repainted.
    emit repaintCanvas();
905 906 907 908 909 910 911 912 913 914 915 916 917
}

void MainWindow::_hideAllHilDockWidgets(void)
{
    foreach(QDockWidget* dockWidget, _mapUasId2HilDockWidget) {
        dockWidget->setVisible(false);
    }
}

void MainWindow::_hideAllDockWidgets(void)
{
    foreach(QDockWidget* dockWidget, _mapName2DockWidget) {
        dockWidget->setVisible(false);
918
    }
919
    _hideAllHilDockWidgets();
920
}
921 922

void MainWindow::_showDockWidgetAction(bool show)
923
{
924 925 926
    QAction* action = dynamic_cast<QAction*>(QObject::sender());
    Q_ASSERT(action);
    _showDockWidget(action->data().toString(), show);
927
}
928

929

930
void MainWindow::loadAnalyzeView()
931
{
932
    if (_currentView != VIEW_ANALYZE)
933
    {
934
        _storeCurrentViewState();
935 936
        _currentView = VIEW_ANALYZE;
        _ui.actionAnalyze->setChecked(true);
937
        _loadCurrentViewState();
938 939 940
    }
}

941
void MainWindow::loadPlanView()
942
{
943 944 945 946 947 948
    if (_currentView != VIEW_MISSIONEDITOR)
    {
        _storeCurrentViewState();
        _currentView = VIEW_MISSIONEDITOR;
        _ui.actionPlan->setChecked(true);
        _loadCurrentViewState();
Don Gagne's avatar
Don Gagne committed
949 950 951
    }
}

952
void MainWindow::loadSetupView()
953
{
954
    if (_currentView != VIEW_SETUP)
955
    {
956 957
        _storeCurrentViewState();
        _currentView = VIEW_SETUP;
dogmaphobic's avatar
dogmaphobic committed
958
        _ui.actionSetup->setChecked(true);
959
        _loadCurrentViewState();
960 961 962
    }
}

963
void MainWindow::loadFlightView()
964
{
965
    if (_currentView != VIEW_FLIGHT)
966
    {
967 968
        _storeCurrentViewState();
        _currentView = VIEW_FLIGHT;
969
        _ui.actionFlight->setChecked(true);
970
        _loadCurrentViewState();
971 972 973
    }
}

974 975
void MainWindow::loadSimulationView()
{
976
    if (_currentView != VIEW_SIMULATION)
977
    {
978 979
        _storeCurrentViewState();
        _currentView = VIEW_SIMULATION;
dogmaphobic's avatar
dogmaphobic committed
980
        _ui.actionSimulationView->setChecked(true);
981
        _loadCurrentViewState();
982 983 984
    }
}

Don Gagne's avatar
Don Gagne committed
985
/// @brief Hides the spash screen if it is currently being shown
986
void MainWindow::hideSplashScreen(void)
Don Gagne's avatar
Don Gagne committed
987 988 989 990 991 992 993
{
    if (_splashScreen) {
        _splashScreen->hide();
        _splashScreen = NULL;
    }
}

994 995
void MainWindow::manageLinks()
{
dogmaphobic's avatar
dogmaphobic committed
996
    SettingsDialog settings(this, SettingsDialog::ShowCommLinks);
997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019
    settings.exec();
}

/// @brief Saves the last used connection
void MainWindow::saveLastUsedConnection(const QString connection)
{
    QSettings settings;
    QString key(MAIN_SETTINGS_GROUP);
    key += "/LAST_CONNECTION";
    settings.setValue(key, connection);
}

/// @brief Restore (and connects) the last used connection (if any)
void MainWindow::restoreLastUsedConnection()
{
    // TODO This should check and see of the port/whatever is present
    // first. That is, if the last connection was to a PX4 on some serial
    // port, it should check and see if the port is present before making
    // the connection.
    QSettings settings;
    QString key(MAIN_SETTINGS_GROUP);
    key += "/LAST_CONNECTION";
    if(settings.contains(key)) {
1020
        QString connection = settings.value(key).toString();
1021
        // Create a link for it
1022
        LinkManager::instance()->createConnectedLink(connection);
1023 1024
    }
}
Don Gagne's avatar
Don Gagne committed
1025

1026 1027 1028 1029 1030
void MainWindow::_linkStateChange(LinkInterface*)
{
    emit repaintCanvas();
}

1031
#ifdef QGC_MOUSE_ENABLED_LINUX
1032 1033 1034
bool MainWindow::x11Event(XEvent *event)
{
    emit x11EventOccured(event);
1035
    return false;
1036
}
1037
#endif // QGC_MOUSE_ENABLED_LINUX
1038 1039 1040 1041 1042 1043 1044

#ifdef UNITTEST_BUILD
void MainWindow::_showQmlTestWidget(void)
{
    new QmlTestWidget();
}
#endif