MainWindow.cc 31.9 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>
36

37
#include <QQuickView>
38
#include <QDesktopWidget>
39 40
#include <QScreen>
#include <QDesktopServices>
41

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

#ifndef __mobile__
#include "HILDockWidget.h"
#endif

#ifndef __ios__
#include "SerialLink.h"
#endif
76

77 78 79 80
#ifdef UNITTEST_BUILD
#include "QmlControls/QmlTestWidget.h"
#endif

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


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

89
const char* MainWindow::_mavlinkDockWidgetName = "MAVLINK_INSPECTOR_DOCKWIDGET";
Don Gagne's avatar
Don Gagne committed
90
const char* MainWindow::_customCommandWidgetName = "CUSTOM_COMMAND_DOCKWIDGET";
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";
96
const char* MainWindow::_hilDockWidgetName = "HIL_DOCKWIDGET";
97

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

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

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

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

Don Gagne's avatar
Don Gagne committed
119 120 121
/// @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
122 123 124
MainWindow::MainWindow(QSplashScreen* splashScreen)
    : _autoReconnect(false)
    , _lowPowerMode(false)
125
    , _showStatusBar(false)
dogmaphobic's avatar
dogmaphobic committed
126 127 128 129 130
    , _centerStackActionGroup(new QActionGroup(this))
    , _centralLayout(NULL)
    , _currentViewWidget(NULL)
    , _splashScreen(splashScreen)
    , _currentView(VIEW_SETUP)
131
{
Don Gagne's avatar
Don Gagne committed
132 133
    Q_ASSERT(_instance == NULL);
    _instance = this;
134

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

139 140 141 142
    // 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
143
    // Setup user interface
144
    loadSettings();
145
    emit initStatusChanged(tr("Setting up user interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
dogmaphobic's avatar
dogmaphobic committed
146 147
    _ui.setupUi(this);
    // Make sure tool bar elements all fit before changing minimum width
dogmaphobic's avatar
dogmaphobic committed
148
    setMinimumWidth(1008);
dogmaphobic's avatar
dogmaphobic committed
149
    configureWindowName();
150

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

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

165 166 167
#ifdef UNITTEST_BUILD
    QAction* qmlTestAction = new QAction("Test QML palette and controls", NULL);
    connect(qmlTestAction, &QAction::triggered, this, &MainWindow::_showQmlTestWidget);
168
    _ui.menuWidgets->addAction(qmlTestAction);
169
#endif
170

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

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

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

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

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

206 207 208 209
    // 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);

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

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

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

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

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

264
    connect(_ui.actionStatusBar,  &QAction::triggered, this, &MainWindow::showStatusBarCallback);
265

266 267
    // 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
268
    _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Meta+1", 0));
269 270 271
    _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));
dogmaphobic's avatar
dogmaphobic committed
272
    _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Meta+Return", 0));
273
#else
dogmaphobic's avatar
dogmaphobic committed
274
    _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Ctrl+1", 0));
Gus Grubba's avatar
Gus Grubba committed
275
    _ui.actionPlan->setShortcut(QApplication::translate("MainWindow", "Ctrl+2", 0));
276
    _ui.actionFlight->setShortcut(QApplication::translate("MainWindow", "Ctrl+3", 0));
Gus Grubba's avatar
Gus Grubba committed
277
    _ui.actionAnalyze->setShortcut(QApplication::translate("MainWindow", "Ctrl+4", 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);
344
    _ui.menuWidgets->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
        { _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 },
394
        { _hilDockWidgetName,               "HIL Config",               Qt::LeftDockWidgetArea },
395 396
    };
    static const size_t cDockWidgetInfo = sizeof(rgDockWidgetInfo) / sizeof(rgDockWidgetInfo[0]);
397

398 399 400
    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 */);
401
    }
402
}
403

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

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

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

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

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

444 445
/// Shows or hides the specified dock widget, creating if necessary
void MainWindow::_showDockWidget(const QString& name, bool show)
446
{
447
    if (!_mapName2DockWidget.contains(name)) {
Don Gagne's avatar
Don Gagne committed
448 449
        // 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.
450 451
        return;
    }
452

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

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

462
    dockWidget->setVisible(show);
463

464 465 466 467 468 469 470 471 472
    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
473

474
    QWidget* widget = NULL;
475

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

498 499 500 501 502
    if (widget) {
        QDockWidget* dockWidget = _mapName2DockWidget[widgetName];
        Q_CHECK_PTR(dockWidget);
        widget->setParent(dockWidget);
        dockWidget->setWidget(widget);
503 504
    }
}
505

506
void MainWindow::fullScreenActionItemCallback(bool)
507
{
dogmaphobic's avatar
dogmaphobic committed
508
    _ui.actionNormal->setChecked(false);
509 510
}

511
void MainWindow::normalActionItemCallback(bool)
512
{
dogmaphobic's avatar
dogmaphobic committed
513
    _ui.actionFullscreen->setChecked(false);
514 515
}

516 517 518 519 520 521
void MainWindow::showStatusBarCallback(bool checked)
{
    _showStatusBar = checked;
    checked ? statusBar()->show() : statusBar()->hide();
}

522 523
void MainWindow::closeEvent(QCloseEvent *event)
{
524
    // Disallow window close if there are active connections
525
    if (LinkManager::instance()->anyConnectedLinks()) {
dogmaphobic's avatar
dogmaphobic committed
526 527 528 529 530 531
        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);
532 533 534 535 536 537 538 539 540 541
		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;
542 543
    }

544 545
    // This will process any remaining flight log save dialogs
    qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents);
546
    
547
    // Should not be any active connections
548 549
    Q_ASSERT(!LinkManager::instance()->anyConnectedLinks());
    
550
    _storeCurrentViewState();
551
    storeSettings();
552
    event->accept();
553 554 555 556
}

void MainWindow::loadSettings()
{
557
    // Why the screaming?
558
    QSettings settings;
559
    settings.beginGroup(MAIN_SETTINGS_GROUP);
560 561 562
    _autoReconnect  = settings.value("AUTO_RECONNECT",      _autoReconnect).toBool();
    _lowPowerMode   = settings.value("LOW_POWER_MODE",      _lowPowerMode).toBool();
    _showStatusBar  = settings.value("SHOW_STATUSBAR",      _showStatusBar).toBool();
563 564 565 566 567 568
    settings.endGroup();
}

void MainWindow::storeSettings()
{
    QSettings settings;
569
    settings.beginGroup(MAIN_SETTINGS_GROUP);
570 571 572
    settings.setValue("AUTO_RECONNECT",     _autoReconnect);
    settings.setValue("LOW_POWER_MODE",     _lowPowerMode);
    settings.setValue("SHOW_STATUSBAR",     _showStatusBar);
573
    settings.endGroup();
dogmaphobic's avatar
dogmaphobic committed
574
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
575
	
576
    // Save the last current view in any case
577
    settings.setValue("CURRENT_VIEW", _currentView);
578
    settings.setValue(_getWindowStateKey(), saveState());
579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602
}

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
603
    _autoReconnect = enabled;
604 605 606 607 608 609 610 611 612
}

/**
* @brief Create all actions associated to the main window
*
**/
void MainWindow::connectCommonActions()
{
    // Bind together the perspective actions
dogmaphobic's avatar
dogmaphobic committed
613
    QActionGroup* perspectives = new QActionGroup(_ui.menuPerspectives);
614 615 616
    perspectives->addAction(_ui.actionAnalyze);
    perspectives->addAction(_ui.actionFlight);
    perspectives->addAction(_ui.actionPlan);
dogmaphobic's avatar
dogmaphobic committed
617
    perspectives->addAction(_ui.actionSetup);
618 619 620
    perspectives->setExclusive(true);

    // Mark the right one as selected
621
    if (_currentView == VIEW_ANALYZE)
622
    {
623 624
        _ui.actionAnalyze->setChecked(true);
        _ui.actionAnalyze->activate(QAction::Trigger);
625
    }
626
    if (_currentView == VIEW_FLIGHT)
627
    {
628 629
        _ui.actionFlight->setChecked(true);
        _ui.actionFlight->activate(QAction::Trigger);
630
    }
631
    if (_currentView == VIEW_MISSIONEDITOR)
632
    {
633 634 635
        _ui.actionPlan->setChecked(true);
        _ui.actionPlan->activate(QAction::Trigger);
    }
636
    if (_currentView == VIEW_SETUP)
637
    {
dogmaphobic's avatar
dogmaphobic committed
638 639
        _ui.actionSetup->setChecked(true);
        _ui.actionSetup->activate(QAction::Trigger);
640
    }
641 642

    // Connect actions from ui
dogmaphobic's avatar
dogmaphobic committed
643
    connect(_ui.actionAdd_Link, SIGNAL(triggered()), this, SLOT(manageLinks()));
644 645

    // Connect internal actions
646
    connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleAdded, this, &MainWindow::_vehicleAdded);
647 648

    // Views actions
Don Gagne's avatar
Don Gagne committed
649 650 651
    connect(_ui.actionFlight,           SIGNAL(triggered()), this, SLOT(loadFlightView()));
    connect(_ui.actionAnalyze,          SIGNAL(triggered()), this, SLOT(loadAnalyzeView()));
    connect(_ui.actionPlan,             SIGNAL(triggered()), this, SLOT(loadPlanView()));
652
    
653
    // Help Actions
dogmaphobic's avatar
dogmaphobic committed
654 655 656
    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()));
657 658

    // Audio output
dogmaphobic's avatar
dogmaphobic committed
659 660 661
    _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)));
662 663

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

dogmaphobic's avatar
dogmaphobic committed
666
    // Update Tool Bar
667
    _mainToolBar->setCurrentView(_currentView);
668 669
}

Don Gagne's avatar
Don Gagne committed
670
void MainWindow::_openUrl(const QString& url, const QString& errorMessage)
671
{
Don Gagne's avatar
Don Gagne committed
672
    if(!QDesktopServices::openUrl(QUrl(url))) {
dogmaphobic's avatar
dogmaphobic committed
673 674 675 676
        QMessageBox::critical(
            this,
            tr("Could not open information in browser"),
            errorMessage);
677 678 679
    }
}

Don Gagne's avatar
Don Gagne committed
680 681
void MainWindow::showHelp()
{
dogmaphobic's avatar
dogmaphobic committed
682 683 684
    _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
685 686
}

687 688
void MainWindow::showCredits()
{
dogmaphobic's avatar
dogmaphobic committed
689 690 691
    _openUrl(
        "http://qgroundcontrol.org/credits",
        tr("To get to the credits, please open http://qgroundcontrol.org/credits in a browser."));
692 693 694 695
}

void MainWindow::showRoadMap()
{
dogmaphobic's avatar
dogmaphobic committed
696 697 698
    _openUrl(
        "http://qgroundcontrol.org/dev/roadmap",
        tr("To get to the online help, please open http://qgroundcontrol.org/roadmap in a browser."));
699 700 701 702
}

void MainWindow::showSettings()
{
dogmaphobic's avatar
dogmaphobic committed
703
    SettingsDialog settings(this);
Don Gagne's avatar
Don Gagne committed
704
    settings.exec();
705 706
}

707 708
void MainWindow::commsWidgetDestroyed(QObject *obj)
{
709 710
    // 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
711
    if (_commsWidgetList.contains(obj))
712
    {
dogmaphobic's avatar
dogmaphobic committed
713
        _commsWidgetList.removeOne(obj);
714 715
    }
}
716

717
void MainWindow::_vehicleAdded(Vehicle* vehicle)
718
{
719
    connect(vehicle->uas(), SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)), this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
720

721 722 723
    if (!linechartWidget)
    {
        linechartWidget = new Linecharts(this);
724
        linechartWidget->setVisible(false);
725
    }
726

727
    linechartWidget->addSource(mavlinkDecoder);
728
    if (_analyzeView != linechartWidget)
729
    {
730
        _analyzeView = linechartWidget;
731
    }
732 733
}

734 735
/// Stores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_storeCurrentViewState(void)
736
{
Don Gagne's avatar
Don Gagne committed
737
#ifndef __mobile__
738 739 740 741 742 743 744 745 746 747 748
    // 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;
        }
749
    }
dogmaphobic's avatar
dogmaphobic committed
750
    settings.setValue(_getWindowStateKey() + "WIDGETS", widgetNames);
Don Gagne's avatar
Don Gagne committed
751
#endif
dogmaphobic's avatar
dogmaphobic committed
752 753
    settings.setValue(_getWindowStateKey(), saveState());
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
754 755
}

756 757
/// Restores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_loadCurrentViewState(void)
758
{
Don Gagne's avatar
Don Gagne committed
759
    QWidget* centerView = NULL;
760
    QString defaultWidgets;
761

762
    switch (_currentView) {
763
        case VIEW_SETUP:
764 765
            _buildSetupView();
            centerView = _setupView;
766
            break;
767

768 769 770
        case VIEW_ANALYZE:
            _buildAnalyzeView();
            centerView = _analyzeView;
771
            defaultWidgets = "PARAMETER_INTERFACE_DOCKWIDGET,FILE_VIEW_DOCKWIDGET";
772
            break;
773

774
        case VIEW_FLIGHT:
775 776
            _buildFlightView();
            centerView = _flightView;
777
            defaultWidgets = "COMMUNICATION_CONSOLE_DOCKWIDGET,UAS_INFO_INFOVIEW_DOCKWIDGET";
778
            break;
779

Don Gagne's avatar
Don Gagne committed
780 781 782 783 784
        case VIEW_MISSIONEDITOR:
            _buildMissionEditorView();
            centerView = _missionEditorView;
            break;

785 786
        default:
            Q_ASSERT(false);
787
            break;
788
    }
789

790 791 792 793 794 795 796 797
    // Remove old view
    if (_currentViewWidget) {
        _currentViewWidget->setVisible(false);
        Q_ASSERT(_centralLayout->count() == 1);
        QLayoutItem *child = _centralLayout->takeAt(0);
        Q_ASSERT(child);
        delete child;
    }
798

799 800 801 802 803
    // Add the new one
    Q_ASSERT(centerView);
    Q_ASSERT(_centralLayout->count() == 0);
    _currentViewWidget = centerView;
    _centralLayout->addWidget(_currentViewWidget);
804
    _centralLayout->setContentsMargins(0, 0, 0, 0);
805
    _currentViewWidget->setVisible(true);
806

807 808 809
    // Hide all widgets from previous view
    _hideAllDockWidgets();

Don Gagne's avatar
Don Gagne committed
810
#ifndef __mobile__
811
    // Restore the widgets for the new view
dogmaphobic's avatar
dogmaphobic committed
812
    QString widgetNames = settings.value(_getWindowStateKey() + "WIDGETS", defaultWidgets).toString();
813
    qDebug() << widgetNames;
814 815 816 817 818
    if (!widgetNames.isEmpty()) {
        QStringList split = widgetNames.split(",");
        foreach (QString widgetName, split) {
            Q_ASSERT(!widgetName.isEmpty());
            _showDockWidget(widgetName, true);
819 820
        }
    }
Don Gagne's avatar
Don Gagne committed
821
#endif
822

dogmaphobic's avatar
dogmaphobic committed
823 824
    if (settings.contains(_getWindowStateKey())) {
        restoreState(settings.value(_getWindowStateKey()).toByteArray());
825
    }
826

827 828 829
    // 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();
830 831 832 833 834 835
}

void MainWindow::_hideAllDockWidgets(void)
{
    foreach(QDockWidget* dockWidget, _mapName2DockWidget) {
        dockWidget->setVisible(false);
836 837
    }
}
838 839

void MainWindow::_showDockWidgetAction(bool show)
840
{
841 842 843
    QAction* action = dynamic_cast<QAction*>(QObject::sender());
    Q_ASSERT(action);
    _showDockWidget(action->data().toString(), show);
844
}
845

846

847
void MainWindow::loadAnalyzeView()
848
{
849
    if (_currentView != VIEW_ANALYZE)
850
    {
851
        _storeCurrentViewState();
852 853
        _currentView = VIEW_ANALYZE;
        _ui.actionAnalyze->setChecked(true);
854
        _loadCurrentViewState();
855 856 857
    }
}

858
void MainWindow::loadPlanView()
859
{
860 861 862 863 864 865
    if (_currentView != VIEW_MISSIONEDITOR)
    {
        _storeCurrentViewState();
        _currentView = VIEW_MISSIONEDITOR;
        _ui.actionPlan->setChecked(true);
        _loadCurrentViewState();
Don Gagne's avatar
Don Gagne committed
866 867 868
    }
}

869
void MainWindow::loadSetupView()
870
{
871
    if (_currentView != VIEW_SETUP)
872
    {
873 874
        _storeCurrentViewState();
        _currentView = VIEW_SETUP;
dogmaphobic's avatar
dogmaphobic committed
875
        _ui.actionSetup->setChecked(true);
876
        _loadCurrentViewState();
877 878 879
    }
}

880
void MainWindow::loadFlightView()
881
{
882
    if (_currentView != VIEW_FLIGHT)
883
    {
884 885
        _storeCurrentViewState();
        _currentView = VIEW_FLIGHT;
886
        _ui.actionFlight->setChecked(true);
887
        _loadCurrentViewState();
888 889 890
    }
}

Don Gagne's avatar
Don Gagne committed
891
/// @brief Hides the spash screen if it is currently being shown
892
void MainWindow::hideSplashScreen(void)
Don Gagne's avatar
Don Gagne committed
893 894 895 896 897 898 899
{
    if (_splashScreen) {
        _splashScreen->hide();
        _splashScreen = NULL;
    }
}

900 901
void MainWindow::manageLinks()
{
dogmaphobic's avatar
dogmaphobic committed
902
    SettingsDialog settings(this, SettingsDialog::ShowCommLinks);
903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925
    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)) {
926
        QString connection = settings.value(key).toString();
927
        // Create a link for it
928
        LinkManager::instance()->createConnectedLink(connection);
929 930
    }
}
Don Gagne's avatar
Don Gagne committed
931

932 933 934 935 936
void MainWindow::_linkStateChange(LinkInterface*)
{
    emit repaintCanvas();
}

937
#ifdef QGC_MOUSE_ENABLED_LINUX
938 939 940
bool MainWindow::x11Event(XEvent *event)
{
    emit x11EventOccured(event);
941
    return false;
942
}
943
#endif // QGC_MOUSE_ENABLED_LINUX
944 945 946 947 948 949 950

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