Skip to content
Snippets Groups Projects
MainWindow.cc 39.2 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*=====================================================================
    
    QGroundControl Open Source Ground Control Station
    
    
    (c) 2009 - 2013 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
    
    
    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
    #ifndef __mobile__
    
    #include <QGCHilLink.h>
    #include <QGCHilConfiguration.h>
    #include <QGCHilFlightGearConfiguration.h>
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    #include <QQuickView>
    
    #include <QDesktopWidget>
    
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __ios__
    
    #include "SerialLink.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    #include "MAVLinkProtocol.h"
    #include "QGCWaypointListMulti.h"
    #include "MainWindow.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
    #include "JoystickWidget.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    #include "GAudioOutput.h"
    #include "QGCMAVLinkLogPlayer.h"
    
    Don Gagne's avatar
    Don Gagne committed
    #include "SettingsDialog.h"
    
    #include "QGCMapTool.h"
    
    #include "QGCMapDisplay.h"
    
    #include "MAVLinkDecoder.h"
    #include "QGCMAVLinkMessageSender.h"
    #include "QGCRGBDView.h"
    
    #include "QGCDataPlot2D.h"
    #include "Linecharts.h"
    
    #include "QGCTabbedInfoView.h"
    #include "UASRawStatusView.h"
    
    #include "FlightDisplay.h"
    
    #include "SetupView.h"
    
    #include "QGCUASFileViewMulti.h"
    
    Don Gagne's avatar
    Don Gagne committed
    #include "QGCApplication.h"
    
    #include "QGCFileDialog.h"
    
    Don Gagne's avatar
    Don Gagne committed
    #include "QGCMessageBox.h"
    
    #include "QGCDockWidget.h"
    
    #include "MultiVehicleManager.h"
    
    Don Gagne's avatar
    Don Gagne committed
    #include "CustomCommandWidget.h"
    
    #include "HomePositionManager.h"
    
    #ifdef UNITTEST_BUILD
    #include "QmlControls/QmlTestWidget.h"
    #endif
    
    
    #ifdef QGC_OSG_ENABLED
    #include "Q3DWidgetFactory.h"
    #endif
    
    #include "LogCompressor.h"
    
    
    /// The key under which the Main Window settings are saved
    const char* MAIN_SETTINGS_GROUP = "QGC_MAINWINDOW";
    
    
    const char* MainWindow::_uasControlDockWidgetName = "UNMANNED_SYSTEM_CONTROL_DOCKWIDGET";
    const char* MainWindow::_uasListDockWidgetName = "UNMANNED_SYSTEM_LIST_DOCKWIDGET";
    const char* MainWindow::_waypointsDockWidgetName = "WAYPOINT_LIST_DOCKWIDGET";
    const char* MainWindow::_mavlinkDockWidgetName = "MAVLINK_INSPECTOR_DOCKWIDGET";
    const char* MainWindow::_parametersDockWidgetName = "PARAMETER_INTERFACE_DOCKWIDGET";
    
    Don Gagne's avatar
    Don Gagne committed
    const char* MainWindow::_customCommandWidgetName = "CUSTOM_COMMAND_DOCKWIDGET";
    
    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::_hsiDockWidgetName = "HORIZONTAL_SITUATION_INDICATOR_DOCKWIDGET";
    const char* MainWindow::_hdd1DockWidgetName = "HEAD_DOWN_DISPLAY_1_DOCKWIDGET";
    const char* MainWindow::_hdd2DockWidgetName = "HEAD_DOWN_DISPLAY_2_DOCKWIDGET";
    const char* MainWindow::_pfdDockWidgetName = "PRIMARY_FLIGHT_DISPLAY_DOCKWIDGET";
    const char* MainWindow::_hudDockWidgetName = "HEAD_UP_DISPLAY_DOCKWIDGET";
    const char* MainWindow::_uasInfoViewDockWidgetName = "UAS_INFO_INFOVIEW_DOCKWIDGET";
    
    
    Don Gagne's avatar
    Don Gagne committed
    static MainWindow* _instance = NULL;   ///< @brief MainWindow singleton
    
    
    MainWindow* MainWindow::_create(QSplashScreen* splashScreen)
    
    Don Gagne's avatar
    Don Gagne committed
        Q_ASSERT(_instance == NULL);
    
        new MainWindow(splashScreen);
    
    Don Gagne's avatar
    Don Gagne committed
        // _instance is set in constructor
        Q_ASSERT(_instance);
    
    Don Gagne's avatar
    Don Gagne committed
    MainWindow* MainWindow::instance(void)
    
    Don Gagne's avatar
    Don Gagne committed
        return _instance;
    
    void MainWindow::deleteInstance(void)
    {
    
    Don Gagne's avatar
    Don Gagne committed
        delete this;
    
    Don Gagne's avatar
    Don Gagne committed
    /// @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
    MainWindow::MainWindow(QSplashScreen* splashScreen)
        : _autoReconnect(false)
        , _lowPowerMode(false)
    
    dogmaphobic's avatar
    dogmaphobic committed
        , _centerStackActionGroup(new QActionGroup(this))
        , _centralLayout(NULL)
        , _currentViewWidget(NULL)
        , _splashScreen(splashScreen)
        , _currentView(VIEW_SETUP)
    
    Don Gagne's avatar
    Don Gagne committed
        Q_ASSERT(_instance == NULL);
        _instance = this;
    
        if (splashScreen) {
            connect(this, &MainWindow::initStatusChanged, splashScreen, &QSplashScreen::showMessage);
        }
    
        // 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
        // Setup user interface
    
        emit initStatusChanged(tr("Setting up user interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.setupUi(this);
        // Make sure tool bar elements all fit before changing minimum width
    
    dogmaphobic's avatar
    dogmaphobic committed
        setMinimumWidth(1008);
    
    dogmaphobic's avatar
    dogmaphobic committed
        configureWindowName();
    
        // Setup central widget with a layout to hold the views
        _centralLayout = new QVBoxLayout();
    
        _centralLayout->setContentsMargins(0,0,0,0);
    
        centralWidget()->setLayout(_centralLayout);
    
        // Set dock options
        setDockOptions(AnimatedDocks | AllowTabbedDocks | AllowNestedDocks);
        // Setup corners
    
        setCorner(Qt::BottomRightCorner, Qt::BottomDockWidgetArea);
    
    dogmaphobic's avatar
    dogmaphobic committed
        // On Mobile devices, we don't want any main menus at all.
    #ifdef __mobile__
        menuBar()->setNativeMenuBar(false);
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    dogmaphobic's avatar
    dogmaphobic committed
    
    
    #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
        _ui.menuTools->addAction(qmlTestAction);
    
    dogmaphobic's avatar
    dogmaphobic committed
        // Load QML Toolbar
        QDockWidget* widget = new QDockWidget(this);
        widget->setObjectName("ToolBarDockWidget");
        qmlRegisterType<MainToolBar>("QGroundControl.MainToolBar", 1, 0, "MainToolBar");
    
        _mainToolBar = new MainToolBar(widget);
    
    dogmaphobic's avatar
    dogmaphobic committed
        widget->setWidget(_mainToolBar);
        widget->setFeatures(QDockWidget::NoDockWidgetFeatures);
        widget->setTitleBarWidget(new QWidget(this)); // Disables the title bar
        addDockWidget(Qt::TopDockWidgetArea, widget);
    
    dogmaphobic's avatar
    dogmaphobic committed
        // Setup UI state machines
        _centerStackActionGroup->setExclusive(true);
        // Status Bar
    
        setStatusBar(new QStatusBar(this));
    
        statusBar()->setSizeGripEnabled(true);
    
        emit initStatusChanged(tr("Building common widgets."), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
        _buildCommonWidgets();
    
        emit initStatusChanged(tr("Building common actions"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
        // Create actions
        connectCommonActions();
        // Connect user interface devices
    
        emit initStatusChanged(tr("Initializing joystick interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
        joystick = new JoystickInput();
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    #ifdef QGC_MOUSE_ENABLED_WIN
    
        emit initStatusChanged(tr("Initializing 3D mouse interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
        mouseInput = new Mouse3DInput(this);
        mouse = new Mouse6dofInput(mouseInput);
    
    #endif //QGC_MOUSE_ENABLED_WIN
    
    #if QGC_MOUSE_ENABLED_LINUX
    
        emit initStatusChanged(tr("Initializing 3D mouse interface"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
    
        mouse = new Mouse6dofInput(this);
    
        connect(this, SIGNAL(x11EventOccured(XEvent*)), mouse, SLOT(handleX11Event(XEvent*)));
    
    #endif //QGC_MOUSE_ENABLED_LINUX
    
        // 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);
    
    
    dogmaphobic's avatar
    dogmaphobic committed
        if (_autoReconnect)
    
    dogmaphobic's avatar
    dogmaphobic committed
        enableLowPowerMode(_lowPowerMode);
    
        emit initStatusChanged(tr("Restoring last view state"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
        // Restore the window setup
    
        _loadCurrentViewState();
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
        // Restore the window position and size
    
    dogmaphobic's avatar
    dogmaphobic committed
        emit initStatusChanged(tr("Restoring last window size"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
        if (settings.contains(_getWindowGeometryKey()))
    
    dogmaphobic's avatar
    dogmaphobic committed
            restoreGeometry(settings.value(_getWindowGeometryKey()).toByteArray());
    
            QScreen* scr = QApplication::primaryScreen();
            QSize scrSize = scr->availableSize();
            if (scrSize.width() <= 1280)
    
                resize(scrSize.width(), scrSize.height());
    
                int w = scrSize.width()  > 1600 ? 1600 : scrSize.width();
                int h = scrSize.height() >  800 ?  800 : scrSize.height();
                resize(w, h);
    
                move((scrSize.width() - w) / 2, (scrSize.height() - h) / 2);
    
        // Make sure the proper fullscreen/normal menu item is checked properly.
        if (isFullScreen())
        {
    
    dogmaphobic's avatar
    dogmaphobic committed
            _ui.actionFullscreen->setChecked(true);
            _ui.actionNormal->setChecked(false);
    
    dogmaphobic's avatar
    dogmaphobic committed
            _ui.actionFullscreen->setChecked(false);
            _ui.actionNormal->setChecked(true);
    
        }
    
        // And that they will stay checked properly after user input
    
        connect(_ui.actionFullscreen, &QAction::triggered, this, &MainWindow::fullScreenActionItemCallback);
        connect(_ui.actionNormal,     &QAction::triggered, this, &MainWindow::normalActionItemCallback);
    
        connect(_ui.actionStatusBar,  &QAction::triggered, this, &MainWindow::showStatusBarCallback);
    
        // 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
        _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Meta+1", 0));
    
        _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
        _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Meta+5", 0));
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Meta+Return", 0));
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.actionSetup->setShortcut(QApplication::translate("MainWindow", "Ctrl+1", 0));
    
    Gus Grubba's avatar
    Gus Grubba committed
        _ui.actionPlan->setShortcut(QApplication::translate("MainWindow", "Ctrl+2", 0));
    
        _ui.actionFlight->setShortcut(QApplication::translate("MainWindow", "Ctrl+3", 0));
    
    Gus Grubba's avatar
    Gus Grubba committed
        _ui.actionAnalyze->setShortcut(QApplication::translate("MainWindow", "Ctrl+4", 0));
    
    Don Gagne's avatar
    Don Gagne committed
        _ui.actionSimulationView->setShortcut(QApplication::translate("MainWindow", "Ctrl+5", 0));
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.actionFullscreen->setShortcut(QApplication::translate("MainWindow", "Ctrl+Return", 0));
    
        connect(&windowNameUpdateTimer, SIGNAL(timeout()), this, SLOT(configureWindowName()));
        windowNameUpdateTimer.start(15000);
    
        emit initStatusChanged(tr("Done"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
            _ui.actionStatusBar->setChecked(_showStatusBar);
            showStatusBarCallback(_showStatusBar);
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifdef __mobile__
    
            menuBar()->hide();
    #endif
    
    dogmaphobic's avatar
    dogmaphobic committed
    #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
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
            joystick->shutdown();
            joystick->wait(5000);
    
    Lorenz Meier's avatar
    Lorenz Meier committed
            joystick->deleteLater();
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
        // Delete all UAS objects
    
    dogmaphobic's avatar
    dogmaphobic committed
        for (int i=0;i<_commsWidgetList.size();i++)
    
    dogmaphobic's avatar
    dogmaphobic committed
            _commsWidgetList[i]->deleteLater();
    
    Don Gagne's avatar
    Don Gagne committed
        _instance = NULL;
    
    }
    
    void MainWindow::resizeEvent(QResizeEvent * event)
    {
        QMainWindow::resizeEvent(event);
    }
    
    
    dogmaphobic's avatar
    dogmaphobic committed
    QString MainWindow::_getWindowStateKey()
    
    	return QString::number(_currentView)+"_windowstate_";
    
    dogmaphobic's avatar
    dogmaphobic committed
    QString MainWindow::_getWindowGeometryKey()
    
    void MainWindow::_createDockWidget(const QString& title, const QString& name, Qt::DockWidgetArea area, QWidget* innerWidget)
    {
        Q_ASSERT(!_mapName2DockWidget.contains(name));
    
        // 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
        _ui.menuTools->addAction(action);
    
    	
    	// 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);
    	}
    	
    
        _mapName2DockWidget[name] = dockWidget;
        _mapDockWidget2Action[dockWidget] = action;
        addDockWidget(area, dockWidget);
    }
    
    void MainWindow::_buildCommonWidgets(void)
    
    {
        // Add generic MAVLink decoder
    
    dogmaphobic's avatar
    dogmaphobic committed
        // TODO: This is never deleted
    
        mavlinkDecoder = new MAVLinkDecoder(MAVLinkProtocol::instance(), this);
    
    John Tapsell's avatar
    John Tapsell committed
        connect(mavlinkDecoder, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)),
                          this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
    
    dogmaphobic's avatar
    dogmaphobic committed
        // TODO: Make this optional with a preferences setting or under a "View" menu
    
    Don Gagne's avatar
    Don Gagne committed
        logPlayer = new QGCMAVLinkLogPlayer(statusBar());
    
        statusBar()->addPermanentWidget(logPlayer);
    
        // 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.
    
        struct DockWidgetInfo {
            const char* name;
            const char* title;
            Qt::DockWidgetArea area;
        };
    
        static const struct DockWidgetInfo rgDockWidgetInfo[] = {
            { _uasControlDockWidgetName,        "Control",                  Qt::LeftDockWidgetArea },
            { _uasListDockWidgetName,           "Unmanned Systems",         Qt::RightDockWidgetArea },
            { _waypointsDockWidgetName,         "Mission Plan",             Qt::BottomDockWidgetArea },
            { _mavlinkDockWidgetName,           "MAVLink Inspector",        Qt::RightDockWidgetArea },
    
            { _parametersDockWidgetName,        "Parameter Editor",			Qt::RightDockWidgetArea },
    
    Don Gagne's avatar
    Don Gagne committed
            { _customCommandWidgetName,         "Custom Command",			Qt::RightDockWidgetArea },
    
            { _filesDockWidgetName,             "Onboard Files",            Qt::RightDockWidgetArea },
            { _uasStatusDetailsDockWidgetName,  "Status Details",           Qt::RightDockWidgetArea },
            { _mapViewDockWidgetName,           "Map view",                 Qt::RightDockWidgetArea },
            { _hsiDockWidgetName,               "Horizontal Situation",     Qt::BottomDockWidgetArea },
            { _hdd1DockWidgetName,              "Flight Display",           Qt::RightDockWidgetArea },
            { _hdd2DockWidgetName,              "Actuator Status",          Qt::RightDockWidgetArea },
            { _pfdDockWidgetName,               "Primary Flight Display",   Qt::RightDockWidgetArea },
            { _hudDockWidgetName,               "Video Downlink",           Qt::RightDockWidgetArea },
            { _uasInfoViewDockWidgetName,       "Info View",                Qt::LeftDockWidgetArea },
        };
        static const size_t cDockWidgetInfo = sizeof(rgDockWidgetInfo) / sizeof(rgDockWidgetInfo[0]);
    
        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 */);
    
    void MainWindow::_buildPlanView(void)
    
        if (!_planView) {
            _planView = new QGCMapTool(this);
            _planView->setVisible(false);
    
    void MainWindow::_buildExperimentalPlanView(void)
    
        if (!_experimentalPlanView) {
            _experimentalPlanView = new QGCMapDisplay(this);
            _experimentalPlanView->setVisible(false);
        }
    }
    
    void MainWindow::_buildFlightView(void)
    {
        if (!_flightView) {
    
            _flightView = new FlightDisplay(this);
    
            _flightView->setVisible(false);
    
    void MainWindow::_buildSetupView(void)
    
        if (!_setupView) {
            _setupView = new SetupView(this);
            _setupView->setVisible(false);
        }
    
    void MainWindow::_buildAnalyzeView(void)
    
        if (!_analyzeView) {
            _analyzeView = new QGCDataPlot2D(this);
            _analyzeView->setVisible(false);
    
    void MainWindow::_buildSimView(void)
    {
        if (!_simView) {
            _simView = new QGCMapTool(this);
            _simView->setVisible(false);
        }
    
    /// Shows or hides the specified dock widget, creating if necessary
    void MainWindow::_showDockWidget(const QString& name, bool show)
    
        if (!_mapName2DockWidget.contains(name)) {
    
    Don Gagne's avatar
    Don Gagne committed
            // 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.
    
        // Create the inner widget if we need to
        if (!_mapName2DockWidget[name]->widget()) {
            _createInnerDockWidget(name);
    
    
        Q_ASSERT(_mapName2DockWidget.contains(name));
        QDockWidget* dockWidget = _mapName2DockWidget[name];
        Q_ASSERT(dockWidget);
    
        dockWidget->setVisible(show);
    
        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
    
        QWidget* widget = NULL;
    
        if (widgetName == _uasControlDockWidgetName) {
            widget = new UASControlWidget(this);
        } else if (widgetName == _uasListDockWidgetName) {
            widget = new UASListWidget(this);
        } else if (widgetName == _waypointsDockWidgetName) {
            widget = new QGCWaypointListMulti(this);
        } else if (widgetName == _mavlinkDockWidgetName) {
            widget = new QGCMAVLinkInspector(MAVLinkProtocol::instance(),this);
        } else if (widgetName == _parametersDockWidgetName) {
    
            widget = new ParameterEditorWidget(this);
    
    Don Gagne's avatar
    Don Gagne committed
        } else if (widgetName == _customCommandWidgetName) {
            widget = new CustomCommandWidget(this);
    
        } 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 == _hsiDockWidgetName) {
            widget = new HSIDisplay(this);
        } else if (widgetName == _hdd1DockWidgetName) {
    
            QStringList acceptList;
            acceptList.append("-3.3,ATTITUDE.roll,rad,+3.3,s");
            acceptList.append("-3.3,ATTITUDE.pitch,deg,+3.3,s");
            acceptList.append("-3.3,ATTITUDE.yaw,deg,+3.3,s");
    
            HDDisplay *hddisplay = new HDDisplay(acceptList,"Flight Display",this);
            hddisplay->addSource(mavlinkDecoder);
    
            widget = hddisplay;
        } else if (widgetName == _hdd2DockWidgetName) {
    
            QStringList acceptList;
            acceptList.append("0,RAW_PRESSURE.pres_abs,hPa,65500");
            HDDisplay *hddisplay = new HDDisplay(acceptList,"Actuator Status",this);
    
            widget = hddisplay;
        } else if (widgetName == _pfdDockWidgetName) {
    
            widget = new FlightDisplay(this);
    
        } else if (widgetName == _hudDockWidgetName) {
            widget = new HUD(320,240,this);
        } else if (widgetName == _uasInfoViewDockWidgetName) {
    
            QGCTabbedInfoView* pInfoView = new QGCTabbedInfoView(this);
            pInfoView->addSource(mavlinkDecoder);
            widget = pInfoView;
    
        } else {
            qWarning() << "Attempt to create unknown Inner Dock Widget" << widgetName;
    
        if (widget) {
            QDockWidget* dockWidget = _mapName2DockWidget[widgetName];
            Q_CHECK_PTR(dockWidget);
            widget->setParent(dockWidget);
            dockWidget->setWidget(widget);
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
    void MainWindow::_showHILConfigurationWidgets(void)
    
        Vehicle* vehicle = MultiVehicleManager::instance()->activeVehicle();
    
        if (!vehicle) {
    
        UAS* mav = vehicle->uas();
    
        int uasId = mav->getUASID();
    
        if (!_mapUasId2HilDockWidget.contains(uasId)) {
    
            // Create QDockWidget
    
            QGCDockWidget* dockWidget = new QGCDockWidget(tr("HIL Config %1").arg(uasId), NULL, this);
    
            Q_CHECK_PTR(dockWidget);
            dockWidget->setObjectName(tr("HIL_CONFIG_%1").arg(uasId));
            dockWidget->setVisible (false);
    
            // Create inner widget and set it
            QWidget* widget = new QGCHilConfiguration(mav, dockWidget);
    
            widget->setParent(dockWidget);
            dockWidget->setWidget(widget);
    
            _mapUasId2HilDockWidget[uasId] = dockWidget;
    
            addDockWidget(Qt::LeftDockWidgetArea, dockWidget);
        }
    
        if (_currentView == VIEW_SIMULATION) {
            // HIL dock widgets only show up on simulation view
            foreach (QDockWidget* dockWidget, _mapUasId2HilDockWidget) {
                dockWidget->setVisible(true);
            }
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    void MainWindow::fullScreenActionItemCallback(bool)
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.actionNormal->setChecked(false);
    
    void MainWindow::normalActionItemCallback(bool)
    
    dogmaphobic's avatar
    dogmaphobic committed
        _ui.actionFullscreen->setChecked(false);
    
    void MainWindow::showStatusBarCallback(bool checked)
    {
        _showStatusBar = checked;
        checked ? statusBar()->show() : statusBar()->hide();
    }
    
    
    void MainWindow::closeEvent(QCloseEvent *event)
    {
    
        // Disallow window close if there are active connections
    
        if (LinkManager::instance()->anyConnectedLinks()) {
    
    dogmaphobic's avatar
    dogmaphobic committed
            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);
    
    		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;
    
        // This will process any remaining flight log save dialogs
        qgcApp()->processEvents(QEventLoop::ExcludeUserInputEvents);
    
        // Should not be any active connections
    
        Q_ASSERT(!LinkManager::instance()->anyConnectedLinks());
        
    
        _storeCurrentViewState();
    
        HomePositionManager::instance()->storeSettings();
    
        event->accept();
    
    }
    
    void MainWindow::loadSettings()
    {
    
        settings.beginGroup(MAIN_SETTINGS_GROUP);
    
        _autoReconnect  = settings.value("AUTO_RECONNECT",      _autoReconnect).toBool();
        _lowPowerMode   = settings.value("LOW_POWER_MODE",      _lowPowerMode).toBool();
        _showStatusBar  = settings.value("SHOW_STATUSBAR",      _showStatusBar).toBool();
    
        settings.endGroup();
    }
    
    void MainWindow::storeSettings()
    {
        QSettings settings;
    
        settings.beginGroup(MAIN_SETTINGS_GROUP);
    
        settings.setValue("AUTO_RECONNECT",     _autoReconnect);
        settings.setValue("LOW_POWER_MODE",     _lowPowerMode);
        settings.setValue("SHOW_STATUSBAR",     _showStatusBar);
    
        settings.endGroup();
    
    dogmaphobic's avatar
    dogmaphobic committed
        settings.setValue(_getWindowGeometryKey(), saveGeometry());
    
        // Save the last current view in any case
    
        settings.setValue("CURRENT_VIEW", _currentView);
    
        settings.setValue(_getWindowStateKey(), saveState());
    
    }
    
    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
        _autoReconnect = enabled;
    
    }
    
    /**
    * @brief Create all actions associated to the main window
    *
    **/
    void MainWindow::connectCommonActions()
    {
        // Bind together the perspective actions
    
    dogmaphobic's avatar
    dogmaphobic committed
        QActionGroup* perspectives = new QActionGroup(_ui.menuPerspectives);
    
        perspectives->addAction(_ui.actionAnalyze);
        perspectives->addAction(_ui.actionFlight);
    
    dogmaphobic's avatar
    dogmaphobic committed
        perspectives->addAction(_ui.actionSimulationView);
    
        perspectives->addAction(_ui.actionPlan);
    
    dogmaphobic's avatar
    dogmaphobic committed
        perspectives->addAction(_ui.actionSetup);
    
        perspectives->addAction(_ui.actionExperimentalPlanView);
    
        perspectives->setExclusive(true);
    
        // Mark the right one as selected
    
        if (_currentView == VIEW_ANALYZE)
    
            _ui.actionAnalyze->setChecked(true);
            _ui.actionAnalyze->activate(QAction::Trigger);
    
        if (_currentView == VIEW_FLIGHT)
    
            _ui.actionFlight->setChecked(true);
            _ui.actionFlight->activate(QAction::Trigger);
    
        if (_currentView == VIEW_SIMULATION)
    
    dogmaphobic's avatar
    dogmaphobic committed
            _ui.actionSimulationView->setChecked(true);
            _ui.actionSimulationView->activate(QAction::Trigger);
    
        if (_currentView == VIEW_PLAN)
    
            _ui.actionPlan->setChecked(true);
            _ui.actionPlan->activate(QAction::Trigger);
        }
        if (_currentView == VIEW_EXPERIMENTAL_PLAN)
        {
            _ui.actionExperimentalPlanView->setChecked(true);
            _ui.actionExperimentalPlanView->activate(QAction::Trigger);
    
        if (_currentView == VIEW_SETUP)
    
    dogmaphobic's avatar
    dogmaphobic committed
            _ui.actionSetup->setChecked(true);
            _ui.actionSetup->activate(QAction::Trigger);
    
    
        // Connect actions from ui
    
    dogmaphobic's avatar
    dogmaphobic committed
        connect(_ui.actionAdd_Link, SIGNAL(triggered()), this, SLOT(manageLinks()));
    
    
        // Connect internal actions
    
        connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleAdded, this, &MainWindow::_vehicleAdded);
        connect(MultiVehicleManager::instance(), &MultiVehicleManager::vehicleRemoved, this, &MainWindow::_vehicleRemoved);
    
        connect(_ui.actionFlight, SIGNAL(triggered()), this, SLOT(loadFlightView()));
    
    dogmaphobic's avatar
    dogmaphobic committed
        connect(_ui.actionSimulationView, SIGNAL(triggered()), this, SLOT(loadSimulationView()));
    
        connect(_ui.actionAnalyze, SIGNAL(triggered()), this, SLOT(loadAnalyzeView()));
        connect(_ui.actionPlan, SIGNAL(triggered()), this, SLOT(loadPlanView()));
        connect(_ui.actionExperimentalPlanView, SIGNAL(triggered()), this, SLOT(loadOldPlanView()));
    
    dogmaphobic's avatar
    dogmaphobic committed
        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()));
    
    dogmaphobic's avatar
    dogmaphobic committed
        _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)));
    
    dogmaphobic's avatar
    dogmaphobic committed
        connect(_ui.actionSettings, SIGNAL(triggered()), this, SLOT(showSettings()));
    
    dogmaphobic's avatar
    dogmaphobic committed
        // Update Tool Bar
    
        _mainToolBar->setCurrentView(_currentView);
    
    Don Gagne's avatar
    Don Gagne committed
    void MainWindow::_openUrl(const QString& url, const QString& errorMessage)
    
    Don Gagne's avatar
    Don Gagne committed
        if(!QDesktopServices::openUrl(QUrl(url))) {
    
    dogmaphobic's avatar
    dogmaphobic committed
            QMessageBox::critical(
                this,
                tr("Could not open information in browser"),
                errorMessage);
    
    Don Gagne's avatar
    Don Gagne committed
    void MainWindow::showHelp()
    {
    
    dogmaphobic's avatar
    dogmaphobic committed
        _openUrl(
            "http://qgroundcontrol.org/users/start",
            tr("To get to the online help, please open http://qgroundcontrol.org/user_guide in a browser."));
    
    void MainWindow::showCredits()
    {
    
    dogmaphobic's avatar
    dogmaphobic committed
        _openUrl(
            "http://qgroundcontrol.org/credits",
            tr("To get to the credits, please open http://qgroundcontrol.org/credits in a browser."));
    
    }
    
    void MainWindow::showRoadMap()
    {
    
    dogmaphobic's avatar
    dogmaphobic committed
        _openUrl(
            "http://qgroundcontrol.org/dev/roadmap",
            tr("To get to the online help, please open http://qgroundcontrol.org/roadmap in a browser."));
    
    }
    
    void MainWindow::showSettings()
    {
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
    Don Gagne's avatar
    Don Gagne committed
        SettingsDialog settings(joystick, this);
    
    dogmaphobic's avatar
    dogmaphobic committed
    #else
        SettingsDialog settings(this);
    #endif
    
    Don Gagne's avatar
    Don Gagne committed
        settings.exec();
    
    void MainWindow::commsWidgetDestroyed(QObject *obj)
    {
    
        // 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
        if (_commsWidgetList.contains(obj))
    
    dogmaphobic's avatar
    dogmaphobic committed
            _commsWidgetList.removeOne(obj);
    
    void MainWindow::_vehicleAdded(Vehicle* vehicle)
    
        connect(vehicle->uas(), SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)), this, SIGNAL(valueChanged(int,QString,QString,QVariant,quint64)));
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
        _showHILConfigurationWidgets();
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
        if (!linechartWidget)
        {
            linechartWidget = new Linecharts(this);
    
            linechartWidget->setVisible(false);
    
        linechartWidget->addSource(mavlinkDecoder);
    
        if (_analyzeView != linechartWidget)
    
            _analyzeView = linechartWidget;
    
    void MainWindow::_vehicleRemoved(Vehicle* vehicle)
    
        int vehicleId = vehicle->id();
        
        if (_mapUasId2HilDockWidget.contains(vehicleId)) {
            _mapUasId2HilDockWidget[vehicleId]->deleteLater();
            _mapUasId2HilDockWidget.remove(vehicleId);
    
    /// Stores the state of the toolbar, status bar and widgets associated with the current view
    void MainWindow::_storeCurrentViewState(void)
    
        // HIL dock widgets are dynamic and are not part of the saved state
        _hideAllHilDockWidgets();
        // 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;
            }
    
    dogmaphobic's avatar
    dogmaphobic committed
        settings.setValue(_getWindowStateKey() + "WIDGETS", widgetNames);
        settings.setValue(_getWindowStateKey(), saveState());
        settings.setValue(_getWindowGeometryKey(), saveGeometry());
    
    /// Restores the state of the toolbar, status bar and widgets associated with the current view
    void MainWindow::_loadCurrentViewState(void)
    
    Don Gagne's avatar
    Don Gagne committed
        QWidget* centerView = NULL;
    
        QString defaultWidgets;
    
        switch (_currentView) {
    
            case VIEW_SETUP:
    
                _buildSetupView();
                centerView = _setupView;
    
            case VIEW_ANALYZE:
                _buildAnalyzeView();
                centerView = _analyzeView;
    
                defaultWidgets = "PARAMETER_INTERFACE_DOCKWIDGET,FILE_VIEW_DOCKWIDGET";
    
                _buildFlightView();
                centerView = _flightView;
    
                defaultWidgets = "COMMUNICATION_CONSOLE_DOCKWIDGET,UAS_INFO_INFOVIEW_DOCKWIDGET";
    
            case VIEW_PLAN:
                _buildPlanView();
                centerView = _planView;
    
                defaultWidgets = "WAYPOINT_LIST_DOCKWIDGET";
    
            case VIEW_EXPERIMENTAL_PLAN:
                _buildExperimentalPlanView();
                centerView = _experimentalPlanView;
                defaultWidgets.clear();
                break;
    
    
                _buildSimView();
                centerView = _simView;
                defaultWidgets = "UNMANNED_SYSTEM_CONTROL_DOCKWIDGET,WAYPOINT_LIST_DOCKWIDGET,PARAMETER_INTERFACE_DOCKWIDGET,PRIMARY_FLIGHT_DISPLAY_DOCKWIDGET";
    
            default:
                Q_ASSERT(false);
    
        // Remove old view
        if (_currentViewWidget) {
            _currentViewWidget->setVisible(false);
            Q_ASSERT(_centralLayout->count() == 1);
            QLayoutItem *child = _centralLayout->takeAt(0);
            Q_ASSERT(child);
            delete child;
        }
    
        // Add the new one
        Q_ASSERT(centerView);
        Q_ASSERT(_centralLayout->count() == 0);
        _currentViewWidget = centerView;
        _centralLayout->addWidget(_currentViewWidget);
    
        _centralLayout->setContentsMargins(0, 0, 0, 0);
    
        _currentViewWidget->setVisible(true);
    
        // Hide all widgets from previous view
        _hideAllDockWidgets();
    
        // Restore the widgets for the new view
    
    dogmaphobic's avatar
    dogmaphobic committed
        QString widgetNames = settings.value(_getWindowStateKey() + "WIDGETS", defaultWidgets).toString();
    
        qDebug() << widgetNames;
    
        if (!widgetNames.isEmpty()) {
            QStringList split = widgetNames.split(",");
            foreach (QString widgetName, split) {
                Q_ASSERT(!widgetName.isEmpty());
                _showDockWidget(widgetName, true);
    
    dogmaphobic's avatar
    dogmaphobic committed
        if (settings.contains(_getWindowStateKey())) {
            restoreState(settings.value(_getWindowStateKey()).toByteArray());
    
        // 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
    #ifndef __mobile__
    
        _showHILConfigurationWidgets();
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    
        // 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();
    
    }
    
    void MainWindow::_hideAllHilDockWidgets(void)
    {
        foreach(QDockWidget* dockWidget, _mapUasId2HilDockWidget) {
            dockWidget->setVisible(false);
        }
    }
    
    void MainWindow::_hideAllDockWidgets(void)
    {
        foreach(QDockWidget* dockWidget, _mapName2DockWidget) {