MainWindow.cc 36.2 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 43
#include <QDesktopWidget>

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

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

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

#include "LogCompressor.h"

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

89 90
const char* MainWindow::_waypointsDockWidgetName = "WAYPOINT_LIST_DOCKWIDGET";
const char* MainWindow::_mavlinkDockWidgetName = "MAVLINK_INSPECTOR_DOCKWIDGET";
Don Gagne's avatar
Don Gagne committed
91
const char* MainWindow::_customCommandWidgetName = "CUSTOM_COMMAND_DOCKWIDGET";
92 93 94 95 96 97
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
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);
dogmaphobic's avatar
dogmaphobic committed
168
    _ui.menuTools->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));
Don Gagne's avatar
Don Gagne committed
272
    _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Meta+5", 0));
dogmaphobic's avatar
dogmaphobic committed
273
    _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Meta+Return", 0));
274
#else
dogmaphobic's avatar
dogmaphobic committed
275
    _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Ctrl+1", 0));
Gus Grubba's avatar
Gus Grubba committed
276
    _ui.actionPlan->setShortcut(QApplication::translate("MainWindow", "Ctrl+2", 0));
277
    _ui.actionFlight->setShortcut(QApplication::translate("MainWindow", "Ctrl+3", 0));
Gus Grubba's avatar
Gus Grubba committed
278
    _ui.actionAnalyze->setShortcut(QApplication::translate("MainWindow", "Ctrl+4", 0));
Don Gagne's avatar
Don Gagne committed
279
    _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Ctrl+5", 0));
dogmaphobic's avatar
dogmaphobic committed
280
    _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Ctrl+Return", 0));
281 282
#endif

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

    if (!qgcApp()->runningUnitTests()) {
288 289
        _ui.actionStatusBar->setChecked(_showStatusBar);
        showStatusBarCallback(_showStatusBar);
dogmaphobic's avatar
dogmaphobic committed
290
#ifdef __mobile__
291 292
        menuBar()->hide();
#endif
293
        show();
dogmaphobic's avatar
dogmaphobic committed
294 295 296 297 298 299 300 301 302 303 304 305 306 307 308
#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
309
    }
310 311 312 313 314
}

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

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

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

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

337 338 339
void MainWindow::_createDockWidget(const QString& title, const QString& name, Qt::DockWidgetArea area, QWidget* innerWidget)
{
    Q_ASSERT(!_mapName2DockWidget.contains(name));
340
	
341 342 343 344 345
    // 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
346
    _ui.menuTools->addAction(action);
347 348 349 350 351 352 353 354 355 356 357 358 359
	
	// 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);
	}
	
360 361 362 363 364 365
    _mapName2DockWidget[name] = dockWidget;
    _mapDockWidget2Action[dockWidget] = action;
    addDockWidget(area, dockWidget);
}

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

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

378 379 380
    // 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.
381

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

388 389 390
    static const struct DockWidgetInfo rgDockWidgetInfo[] = {
        { _waypointsDockWidgetName,         "Mission Plan",             Qt::BottomDockWidgetArea },
        { _mavlinkDockWidgetName,           "MAVLink Inspector",        Qt::RightDockWidgetArea },
Don Gagne's avatar
Don Gagne committed
391
        { _customCommandWidgetName,         "Custom Command",			Qt::RightDockWidgetArea },
392 393 394 395 396 397 398
        { _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]);
399

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

406
void MainWindow::_buildPlanView(void)
407
{
408 409 410
    if (!_planView) {
        _planView = new QGCMapTool(this);
        _planView->setVisible(false);
411
    }
412
}
413

Don Gagne's avatar
Don Gagne committed
414 415 416 417 418 419 420 421
void MainWindow::_buildMissionEditorView(void)
{
    if (!_missionEditorView) {
        _missionEditorView = new MissionEditor(this);
        _missionEditorView->setVisible(false);
    }
}

422 423 424
void MainWindow::_buildFlightView(void)
{
    if (!_flightView) {
425
        _flightView = new FlightDisplayView(this);
426
        _flightView->setVisible(false);
427
    }
428 429
}

430
void MainWindow::_buildSetupView(void)
431
{
432 433 434 435
    if (!_setupView) {
        _setupView = new SetupView(this);
        _setupView->setVisible(false);
    }
436 437
}

438
void MainWindow::_buildAnalyzeView(void)
439
{
440 441 442
    if (!_analyzeView) {
        _analyzeView = new QGCDataPlot2D(this);
        _analyzeView->setVisible(false);
443 444
    }
}
445

446 447 448 449 450 451
void MainWindow::_buildSimView(void)
{
    if (!_simView) {
        _simView = new QGCMapTool(this);
        _simView->setVisible(false);
    }
452
}
453

454 455
/// Shows or hides the specified dock widget, creating if necessary
void MainWindow::_showDockWidget(const QString& name, bool show)
456
{
457
    if (!_mapName2DockWidget.contains(name)) {
Don Gagne's avatar
Don Gagne committed
458 459
        // 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.
460 461
        return;
    }
462

463 464 465
    // Create the inner widget if we need to
    if (!_mapName2DockWidget[name]->widget()) {
        _createInnerDockWidget(name);
466
    }
467 468 469 470

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

472
    dockWidget->setVisible(show);
473

474 475 476 477 478 479 480 481 482
    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
483

484
    QWidget* widget = NULL;
485

Don Gagne's avatar
Don Gagne committed
486
    if (widgetName == _waypointsDockWidgetName) {
487 488 489
        widget = new QGCWaypointListMulti(this);
    } else if (widgetName == _mavlinkDockWidgetName) {
        widget = new QGCMAVLinkInspector(MAVLinkProtocol::instance(),this);
Don Gagne's avatar
Don Gagne committed
490 491
    } else if (widgetName == _customCommandWidgetName) {
        widget = new CustomCommandWidget(this);
492 493 494 495 496 497 498
    } else if (widgetName == _filesDockWidgetName) {
        widget = new QGCUASFileViewMulti(this);
    } else if (widgetName == _uasStatusDetailsDockWidgetName) {
        widget = new UASInfoWidget(this);
    } else if (widgetName == _mapViewDockWidgetName) {
        widget = new QGCMapTool(this);
    } else if (widgetName == _pfdDockWidgetName) {
499
        widget = new FlightDisplayWidget(this);
500
    } else if (widgetName == _uasInfoViewDockWidgetName) {
501 502 503
        QGCTabbedInfoView* pInfoView = new QGCTabbedInfoView(this);
        pInfoView->addSource(mavlinkDecoder);
        widget = pInfoView;
504 505
    } else {
        qWarning() << "Attempt to create unknown Inner Dock Widget" << widgetName;
506
    }
507

508 509 510 511 512
    if (widget) {
        QDockWidget* dockWidget = _mapName2DockWidget[widgetName];
        Q_CHECK_PTR(dockWidget);
        widget->setParent(dockWidget);
        dockWidget->setWidget(widget);
513 514
    }
}
515

dogmaphobic's avatar
dogmaphobic committed
516
#ifndef __mobile__
517
void MainWindow::_showHILConfigurationWidgets(void)
518
{
519
    Vehicle* vehicle = MultiVehicleManager::instance()->activeVehicle();
520

521
    if (!vehicle) {
522 523
        return;
    }
524

525
    UAS* mav = vehicle->uas();
526
    Q_ASSERT(mav);
527

528
    int uasId = mav->getUASID();
529

530
    if (!_mapUasId2HilDockWidget.contains(uasId)) {
531

532
        // Create QDockWidget
533
        QGCDockWidget* dockWidget = new QGCDockWidget(tr("HIL Config %1").arg(uasId), NULL, this);
534 535 536
        Q_CHECK_PTR(dockWidget);
        dockWidget->setObjectName(tr("HIL_CONFIG_%1").arg(uasId));
        dockWidget->setVisible (false);
537

538 539
        // Create inner widget and set it
        QWidget* widget = new QGCHilConfiguration(mav, dockWidget);
540

541 542
        widget->setParent(dockWidget);
        dockWidget->setWidget(widget);
543

544
        _mapUasId2HilDockWidget[uasId] = dockWidget;
545

546 547
        addDockWidget(Qt::LeftDockWidgetArea, dockWidget);
    }
548

549 550 551 552 553
    if (_currentView == VIEW_SIMULATION) {
        // HIL dock widgets only show up on simulation view
        foreach (QDockWidget* dockWidget, _mapUasId2HilDockWidget) {
            dockWidget->setVisible(true);
        }
554 555
    }
}
dogmaphobic's avatar
dogmaphobic committed
556
#endif
557

558
void MainWindow::fullScreenActionItemCallback(bool)
559
{
dogmaphobic's avatar
dogmaphobic committed
560
    _ui.actionNormal->setChecked(false);
561 562
}

563
void MainWindow::normalActionItemCallback(bool)
564
{
dogmaphobic's avatar
dogmaphobic committed
565
    _ui.actionFullscreen->setChecked(false);
566 567
}

568 569 570 571 572 573
void MainWindow::showStatusBarCallback(bool checked)
{
    _showStatusBar = checked;
    checked ? statusBar()->show() : statusBar()->hide();
}

574 575
void MainWindow::closeEvent(QCloseEvent *event)
{
576
    // Disallow window close if there are active connections
577
    if (LinkManager::instance()->anyConnectedLinks()) {
dogmaphobic's avatar
dogmaphobic committed
578 579 580 581 582 583
        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);
584 585 586 587 588 589 590 591 592 593
		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;
594 595
    }

596 597
    // This will process any remaining flight log save dialogs
    qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents);
598
    
599
    // Should not be any active connections
600 601
    Q_ASSERT(!LinkManager::instance()->anyConnectedLinks());
    
602
    _storeCurrentViewState();
603
    storeSettings();
604
    event->accept();
605 606 607 608
}

void MainWindow::loadSettings()
{
609
    // Why the screaming?
610
    QSettings settings;
611
    settings.beginGroup(MAIN_SETTINGS_GROUP);
612 613 614
    _autoReconnect  = settings.value("AUTO_RECONNECT",      _autoReconnect).toBool();
    _lowPowerMode   = settings.value("LOW_POWER_MODE",      _lowPowerMode).toBool();
    _showStatusBar  = settings.value("SHOW_STATUSBAR",      _showStatusBar).toBool();
615 616 617 618 619 620
    settings.endGroup();
}

void MainWindow::storeSettings()
{
    QSettings settings;
621
    settings.beginGroup(MAIN_SETTINGS_GROUP);
622 623 624
    settings.setValue("AUTO_RECONNECT",     _autoReconnect);
    settings.setValue("LOW_POWER_MODE",     _lowPowerMode);
    settings.setValue("SHOW_STATUSBAR",     _showStatusBar);
625
    settings.endGroup();
dogmaphobic's avatar
dogmaphobic committed
626
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
627
	
628
    // Save the last current view in any case
629
    settings.setValue("CURRENT_VIEW", _currentView);
630
    settings.setValue(_getWindowStateKey(), saveState());
631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654
}

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
655
    _autoReconnect = enabled;
656 657 658 659 660 661 662 663 664
}

/**
* @brief Create all actions associated to the main window
*
**/
void MainWindow::connectCommonActions()
{
    // Bind together the perspective actions
dogmaphobic's avatar
dogmaphobic committed
665
    QActionGroup* perspectives = new QActionGroup(_ui.menuPerspectives);
666 667
    perspectives->addAction(_ui.actionAnalyze);
    perspectives->addAction(_ui.actionFlight);
dogmaphobic's avatar
dogmaphobic committed
668
    perspectives->addAction(_ui.actionSimulationView);
669
    perspectives->addAction(_ui.actionPlan);
dogmaphobic's avatar
dogmaphobic committed
670
    perspectives->addAction(_ui.actionSetup);
671 672 673
    perspectives->setExclusive(true);

    // Mark the right one as selected
674
    if (_currentView == VIEW_ANALYZE)
675
    {
676 677
        _ui.actionAnalyze->setChecked(true);
        _ui.actionAnalyze->activate(QAction::Trigger);
678
    }
679
    if (_currentView == VIEW_FLIGHT)
680
    {
681 682
        _ui.actionFlight->setChecked(true);
        _ui.actionFlight->activate(QAction::Trigger);
683
    }
684
    if (_currentView == VIEW_SIMULATION)
685
    {
dogmaphobic's avatar
dogmaphobic committed
686 687
        _ui.actionSimulationView->setChecked(true);
        _ui.actionSimulationView->activate(QAction::Trigger);
688
    }
689
    if (_currentView == VIEW_PLAN || _currentView == VIEW_MISSIONEDITOR)
690
    {
691 692 693
        _ui.actionPlan->setChecked(true);
        _ui.actionPlan->activate(QAction::Trigger);
    }
694
    if (_currentView == VIEW_SETUP)
695
    {
dogmaphobic's avatar
dogmaphobic committed
696 697
        _ui.actionSetup->setChecked(true);
        _ui.actionSetup->activate(QAction::Trigger);
698
    }
699 700

    // Connect actions from ui
dogmaphobic's avatar
dogmaphobic committed
701
    connect(_ui.actionAdd_Link, SIGNAL(triggered()), this, SLOT(manageLinks()));
702 703

    // Connect internal actions
704 705
    connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleAdded, this, &MainWindow::_vehicleAdded);
    connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleRemoved, this, &MainWindow::_vehicleRemoved);
706 707

    // Views actions
Don Gagne's avatar
Don Gagne committed
708 709 710 711
    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()));
712 713 714
    
    _ui.actionUseMissionEditor->setChecked(qgcApp()->useNewMissionEditor());
    connect(_ui.actionUseMissionEditor, &QAction::triggered, this, &MainWindow::_setUseMissionEditor);
Don Gagne's avatar
Don Gagne committed
715
    
716
    // Help Actions
dogmaphobic's avatar
dogmaphobic committed
717 718 719
    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()));
720 721

    // Audio output
dogmaphobic's avatar
dogmaphobic committed
722 723 724
    _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)));
725 726

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

dogmaphobic's avatar
dogmaphobic committed
729
    // Update Tool Bar
730
    _mainToolBar->setCurrentView(_currentView);
731 732
}

Don Gagne's avatar
Don Gagne committed
733
void MainWindow::_openUrl(const QString& url, const QString& errorMessage)
734
{
Don Gagne's avatar
Don Gagne committed
735
    if(!QDesktopServices::openUrl(QUrl(url))) {
dogmaphobic's avatar
dogmaphobic committed
736 737 738 739
        QMessageBox::critical(
            this,
            tr("Could not open information in browser"),
            errorMessage);
740 741 742
    }
}

Don Gagne's avatar
Don Gagne committed
743 744
void MainWindow::showHelp()
{
dogmaphobic's avatar
dogmaphobic committed
745 746 747
    _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
748 749
}

750 751
void MainWindow::showCredits()
{
dogmaphobic's avatar
dogmaphobic committed
752 753 754
    _openUrl(
        "http://qgroundcontrol.org/credits",
        tr("To get to the credits, please open http://qgroundcontrol.org/credits in a browser."));
755 756 757 758
}

void MainWindow::showRoadMap()
{
dogmaphobic's avatar
dogmaphobic committed
759 760 761
    _openUrl(
        "http://qgroundcontrol.org/dev/roadmap",
        tr("To get to the online help, please open http://qgroundcontrol.org/roadmap in a browser."));
762 763 764 765
}

void MainWindow::showSettings()
{
dogmaphobic's avatar
dogmaphobic committed
766
    SettingsDialog settings(this);
Don Gagne's avatar
Don Gagne committed
767
    settings.exec();
768 769
}

770 771
void MainWindow::commsWidgetDestroyed(QObject *obj)
{
772 773
    // 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
774
    if (_commsWidgetList.contains(obj))
775
    {
dogmaphobic's avatar
dogmaphobic committed
776
        _commsWidgetList.removeOne(obj);
777 778
    }
}
779

780
void MainWindow::_vehicleAdded(Vehicle* vehicle)
781
{
782
    connect(vehicle->uas(), SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)), this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
783

784
    // HIL
dogmaphobic's avatar
dogmaphobic committed
785
#ifndef __mobile__
786
    _showHILConfigurationWidgets();
dogmaphobic's avatar
dogmaphobic committed
787
#endif
788

789 790 791
    if (!linechartWidget)
    {
        linechartWidget = new Linecharts(this);
792
        linechartWidget->setVisible(false);
793
    }
794

795
    linechartWidget->addSource(mavlinkDecoder);
796
    if (_analyzeView != linechartWidget)
797
    {
798
        _analyzeView = linechartWidget;
799
    }
800 801
}

802
void MainWindow::_vehicleRemoved(Vehicle* vehicle)
803
{
804 805 806 807 808
    int vehicleId = vehicle->id();
    
    if (_mapUasId2HilDockWidget.contains(vehicleId)) {
        _mapUasId2HilDockWidget[vehicleId]->deleteLater();
        _mapUasId2HilDockWidget.remove(vehicleId);
809 810 811
    }
}

812 813
/// Stores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_storeCurrentViewState(void)
814
{
815 816
    // HIL dock widgets are dynamic and are not part of the saved state
    _hideAllHilDockWidgets();
Don Gagne's avatar
Don Gagne committed
817 818
    
#ifndef __mobile__
819 820 821 822 823 824 825 826 827 828 829
    // 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;
        }
830
    }
dogmaphobic's avatar
dogmaphobic committed
831
    settings.setValue(_getWindowStateKey() + "WIDGETS", widgetNames);
Don Gagne's avatar
Don Gagne committed
832
#endif
dogmaphobic's avatar
dogmaphobic committed
833 834
    settings.setValue(_getWindowStateKey(), saveState());
    settings.setValue(_getWindowGeometryKey(), saveGeometry());
835 836
}

837 838
/// Restores the state of the toolbar, status bar and widgets associated with the current view
void MainWindow::_loadCurrentViewState(void)
839
{
Don Gagne's avatar
Don Gagne committed
840
    QWidget* centerView = NULL;
841
    QString defaultWidgets;
842

843
    switch (_currentView) {
844
        case VIEW_SETUP:
845 846
            _buildSetupView();
            centerView = _setupView;
847
            break;
848

849 850 851
        case VIEW_ANALYZE:
            _buildAnalyzeView();
            centerView = _analyzeView;
852
            defaultWidgets = "PARAMETER_INTERFACE_DOCKWIDGET,FILE_VIEW_DOCKWIDGET";
853
            break;
854

855
        case VIEW_FLIGHT:
856 857
            _buildFlightView();
            centerView = _flightView;
858
            defaultWidgets = "COMMUNICATION_CONSOLE_DOCKWIDGET,UAS_INFO_INFOVIEW_DOCKWIDGET";
859
            break;
860

861 862 863
        case VIEW_PLAN:
            _buildPlanView();
            centerView = _planView;
864
            defaultWidgets = "WAYPOINT_LIST_DOCKWIDGET";
865
            break;
866

Don Gagne's avatar
Don Gagne committed
867 868 869 870 871
        case VIEW_MISSIONEDITOR:
            _buildMissionEditorView();
            centerView = _missionEditorView;
            break;

872
        case VIEW_SIMULATION:
873 874
            _buildSimView();
            centerView = _simView;
Don Gagne's avatar
Don Gagne committed
875
            defaultWidgets = "WAYPOINT_LIST_DOCKWIDGET,PARAMETER_INTERFACE_DOCKWIDGET,PRIMARY_FLIGHT_DISPLAY_DOCKWIDGET";
876
            break;
877

878 879
        default:
            Q_ASSERT(false);
880
            break;
881
    }
882

883 884 885 886 887 888 889 890
    // Remove old view
    if (_currentViewWidget) {
        _currentViewWidget->setVisible(false);
        Q_ASSERT(_centralLayout->count() == 1);
        QLayoutItem *child = _centralLayout->takeAt(0);
        Q_ASSERT(child);
        delete child;
    }
891

892 893 894 895 896
    // Add the new one
    Q_ASSERT(centerView);
    Q_ASSERT(_centralLayout->count() == 0);
    _currentViewWidget = centerView;
    _centralLayout->addWidget(_currentViewWidget);
897
    _centralLayout->setContentsMargins(0, 0, 0, 0);
898
    _currentViewWidget->setVisible(true);
899

900 901 902
    // Hide all widgets from previous view
    _hideAllDockWidgets();

Don Gagne's avatar
Don Gagne committed
903
#ifndef __mobile__
904
    // Restore the widgets for the new view
dogmaphobic's avatar
dogmaphobic committed
905
    QString widgetNames = settings.value(_getWindowStateKey() + "WIDGETS", defaultWidgets).toString();
906
    qDebug() << widgetNames;
907 908 909 910 911
    if (!widgetNames.isEmpty()) {
        QStringList split = widgetNames.split(",");
        foreach (QString widgetName, split) {
            Q_ASSERT(!widgetName.isEmpty());
            _showDockWidget(widgetName, true);
912 913
        }
    }
Don Gagne's avatar
Don Gagne committed
914
#endif
915

dogmaphobic's avatar
dogmaphobic committed
916 917
    if (settings.contains(_getWindowStateKey())) {
        restoreState(settings.value(_getWindowStateKey()).toByteArray());
918
    }
919

920 921
    // 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
922
#ifndef __mobile__
923
    _showHILConfigurationWidgets();
dogmaphobic's avatar
dogmaphobic committed
924
#endif
925 926 927 928

    // 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();
929 930 931 932 933 934 935 936 937 938 939 940 941
}

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

void MainWindow::_hideAllDockWidgets(void)
{
    foreach(QDockWidget* dockWidget, _mapName2DockWidget) {
        dockWidget->setVisible(false);
942
    }
943
    _hideAllHilDockWidgets();
944
}
945 946

void MainWindow::_showDockWidgetAction(bool show)
947
{
948 949 950
    QAction* action = dynamic_cast<QAction*>(QObject::sender());
    Q_ASSERT(action);
    _showDockWidget(action->data().toString(), show);
951
}
952

953

954
void MainWindow::loadAnalyzeView()
955
{
956
    if (_currentView != VIEW_ANALYZE)
957
    {
958
        _storeCurrentViewState();
959 960
        _currentView = VIEW_ANALYZE;
        _ui.actionAnalyze->setChecked(true);
961
        _loadCurrentViewState();
962 963 964
    }
}

965
void MainWindow::loadPlanView()
966
{
967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982
    if (qgcApp()->useNewMissionEditor()) {
        if (_currentView != VIEW_MISSIONEDITOR)
        {
            _storeCurrentViewState();
            _currentView = VIEW_MISSIONEDITOR;
            _ui.actionPlan->setChecked(true);
            _loadCurrentViewState();
        }
    } else {
        if (_currentView != VIEW_PLAN)
        {
            _storeCurrentViewState();
            _currentView = VIEW_PLAN;
            _ui.actionPlan->setChecked(true);
            _loadCurrentViewState();
        }
Don Gagne's avatar
Don Gagne committed
983 984 985
    }
}

986
void MainWindow::loadSetupView()
987
{
988
    if (_currentView != VIEW_SETUP)
989
    {
990 991
        _storeCurrentViewState();
        _currentView = VIEW_SETUP;
dogmaphobic's avatar
dogmaphobic committed
992
        _ui.actionSetup->setChecked(true);
993
        _loadCurrentViewState();
994 995 996
    }
}

997
void MainWindow::loadFlightView()
998
{
999
    if (_currentView != VIEW_FLIGHT)
1000
    {
1001 1002
        _storeCurrentViewState();
        _currentView = VIEW_FLIGHT;
1003
        _ui.actionFlight->setChecked(true);
1004
        _loadCurrentViewState();
1005 1006 1007
    }
}

1008 1009
void MainWindow::loadSimulationView()
{
1010
    if (_currentView != VIEW_SIMULATION)
1011
    {
1012 1013
        _storeCurrentViewState();
        _currentView = VIEW_SIMULATION;
dogmaphobic's avatar
dogmaphobic committed
1014
        _ui.actionSimulationView->setChecked(true);
1015
        _loadCurrentViewState();
1016 1017 1018
    }
}

Don Gagne's avatar
Don Gagne committed
1019
/// @brief Hides the spash screen if it is currently being shown
1020
void MainWindow::hideSplashScreen(void)
Don Gagne's avatar
Don Gagne committed
1021 1022 1023 1024 1025 1026 1027
{
    if (_splashScreen) {
        _splashScreen->hide();
        _splashScreen = NULL;
    }
}

1028 1029
void MainWindow::manageLinks()
{
dogmaphobic's avatar
dogmaphobic committed
1030
    SettingsDialog settings(this, SettingsDialog::ShowCommLinks);
1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053
    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)) {
1054
        QString connection = settings.value(key).toString();
1055
        // Create a link for it
1056
        LinkManager::instance()->createConnectedLink(connection);
1057 1058
    }
}
Don Gagne's avatar
Don Gagne committed
1059

1060 1061 1062 1063 1064
void MainWindow::_linkStateChange(LinkInterface*)
{
    emit repaintCanvas();
}

1065
#ifdef QGC_MOUSE_ENABLED_LINUX
1066 1067 1068
bool MainWindow::x11Event(XEvent *event)
{
    emit x11EventOccured(event);
1069
    return false;
1070
}
1071
#endif // QGC_MOUSE_ENABLED_LINUX
1072 1073 1074 1075 1076 1077 1078

#ifdef UNITTEST_BUILD
void MainWindow::_showQmlTestWidget(void)
{
    new QmlTestWidget();
}
#endif
1079 1080 1081 1082 1083

void MainWindow::_setUseMissionEditor(bool checked)
{
    qgcApp()->setUseNewMissionEditor(checked);
}