Skip to content
Snippets Groups Projects
MainWindow.cc 17.3 KiB
Newer Older
  • Learn to ignore specific revisions
  • /****************************************************************************
     *
     *   (c) 2009-2016 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
     *
     * QGroundControl is licensed according to the terms in the file
     * COPYING.md in the root of the source code directory.
     *
     ****************************************************************************/
    
    
    
    /**
     * @file
     *   @brief Implementation of class MainWindow
     *   @author Lorenz Meier <mail@qgroundcontrol.org>
     */
    
    #include <QSettings>
    #include <QNetworkInterface>
    #include <QDebug>
    #include <QTimer>
    #include <QHostInfo>
    
    #include <QQuickView>
    
    #include <QDesktopWidget>
    
    #include <QScreen>
    #include <QDesktopServices>
    
    #include <QDockWidget>
    
    #include <QMenuBar>
    
    Don Gagne's avatar
    Don Gagne committed
    #include <QDialog>
    
    #include "QGC.h"
    #include "MAVLinkProtocol.h"
    #include "MainWindow.h"
    #include "GAudioOutput.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #ifndef __mobile__
    
    #include "QGCMAVLinkLogPlayer.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #endif
    
    #include "MAVLinkDecoder.h"
    
    Don Gagne's avatar
    Don Gagne committed
    #include "QGCApplication.h"
    
    #include "MultiVehicleManager.h"
    
    #include "LogCompressor.h"
    
    #include "UAS.h"
    
    dogmaphobic's avatar
    dogmaphobic committed
    #include "QGCImageProvider.h"
    
    #include "QGCCorePlugin.h"
    
    
    #ifndef __mobile__
    
    #include "Linecharts.h"
    
    #include "QGCUASFileViewMulti.h"
    #include "CustomCommandWidget.h"
    #include "QGCDockWidget.h"
    
    #include "HILDockWidget.h"
    
    Gus Grubba's avatar
    Gus Grubba committed
    #ifndef NO_SERIAL_LINK
    
    #include "SerialLink.h"
    #endif
    
    #ifdef UNITTEST_BUILD
    #include "QmlControls/QmlTestWidget.h"
    #endif
    
    
    /// The key under which the Main Window settings are saved
    const char* MAIN_SETTINGS_GROUP = "QGC_MAINWINDOW";
    
    
    #ifndef __mobile__
    
    enum DockWidgetTypes {
        MAVLINK_INSPECTOR,
        CUSTOM_COMMAND,
    
        ONBOARD_FILES,
    
    DonLakeFlyer's avatar
    DonLakeFlyer committed
        DEPRECATED_WIDGET,
    
        HIL_CONFIG,
    
        ANALYZE
    
    };
    
    static const char *rgDockWidgetNames[] = {
        "MAVLink Inspector",
        "Custom Command",
        "Onboard Files",
    
    DonLakeFlyer's avatar
    DonLakeFlyer committed
        "Deprecated Widget",
    
        "HIL Config",
    
        "Analyze"
    
    };
    
    #define ARRAY_SIZE(ARRAY) (sizeof(ARRAY) / sizeof(ARRAY[0]))
    
    
    static const char* _visibleWidgetsKey = "VisibleWidgets";
    
    Don Gagne's avatar
    Don Gagne committed
    static MainWindow* _instance = NULL;   ///< @brief MainWindow singleton
    
    
    Lorenz Meier's avatar
    Lorenz Meier committed
    MainWindow* MainWindow::_create()
    
    Lorenz Meier's avatar
    Lorenz Meier committed
        new MainWindow();
    
    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.
    
    Lorenz Meier's avatar
    Lorenz Meier committed
    MainWindow::MainWindow()
    
        : _mavlinkDecoder       (NULL)
        , _lowPowerMode         (false)
        , _showStatusBar        (false)
        , _mainQmlWidgetHolder  (NULL)
        , _forceClose           (false)
    
    Don Gagne's avatar
    Don Gagne committed
        _instance = this;
    
        //-- Load fonts
        if(QFontDatabase::addApplicationFont(":/fonts/opensans") < 0) {
            qWarning() << "Could not load /fonts/opensans font";
        }
        if(QFontDatabase::addApplicationFont(":/fonts/opensans-demibold") < 0) {
            qWarning() << "Could not load /fonts/opensans-demibold font";
        }
    
    
        // 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);
    
    
        _mainQmlWidgetHolder = new QGCQmlWidgetHolder(QString(), NULL, this);
        _centralLayout->addWidget(_mainQmlWidgetHolder);
        _mainQmlWidgetHolder->setVisible(true);
    
    
        QQmlEngine::setObjectOwnership(this, QQmlEngine::CppOwnership);
    
        _mainQmlWidgetHolder->setContextPropertyObject("controller", this);
    
        _mainQmlWidgetHolder->setContextPropertyObject("debugMessageModel", AppMessages::getModel());
    
    Don Gagne's avatar
    Don Gagne committed
        _mainQmlWidgetHolder->setSource(QUrl::fromUserInput("qrc:qml/MainWindowHybrid.qml"));
    
    dogmaphobic's avatar
    dogmaphobic committed
        // Image provider
        QQuickImageProvider* pImgProvider = dynamic_cast<QQuickImageProvider*>(qgcApp()->toolbox()->imageProvider());
        _mainQmlWidgetHolder->getEngine()->addImageProvider(QLatin1String("QGCImages"), pImgProvider);
    
    
        setDockOptions(0);
    
        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);
    
        _ui.menuWidgets->addAction(qmlTestAction);
    
        connect(qgcApp()->toolbox()->corePlugin(), &QGCCorePlugin::showAdvancedUIChanged, this, &MainWindow::_showAdvancedUIChanged);
        _showAdvancedUIChanged(qgcApp()->toolbox()->corePlugin()->showAdvancedUI());
    
    
    dogmaphobic's avatar
    dogmaphobic committed
        // Status Bar
    
        setStatusBar(new QStatusBar(this));
    
        statusBar()->setSizeGripEnabled(true);
    
    #ifndef __mobile__
    
        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
    
    #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, &MainWindow::x11EventOccured, mouse, &Mouse6dofInput::handleX11Event);
    
    #endif //QGC_MOUSE_ENABLED_LINUX
    
        // Set low power mode
    
    dogmaphobic's avatar
    dogmaphobic committed
        enableLowPowerMode(_lowPowerMode);
    
        emit initStatusChanged(tr("Restoring last view state"), Qt::AlignLeft | Qt::AlignBottom, QColor(62, 93, 141));
    
    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);
    
        connect(_ui.actionStatusBar,  &QAction::triggered, this, &MainWindow::showStatusBarCallback);
    
        connect(&windowNameUpdateTimer, &QTimer::timeout, this, &MainWindow::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 __macos__
    
    dogmaphobic's avatar
    dogmaphobic committed
            // 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
    
    #ifndef __mobile__
        _loadVisibleWidgetsSettings();
    #endif
    
        //-- Enable message handler display of messages in main window
        UASMessageHandler* msgHandler = qgcApp()->toolbox()->uasMessageHandler();
        if(msgHandler) {
            msgHandler->showErrorsInToolbar();
        }
    
        if (_mavlinkDecoder) {
            // Enforce thread-safe shutdown of the mavlink decoder
            _mavlinkDecoder->finish();
            _mavlinkDecoder->wait(1000);
            _mavlinkDecoder->deleteLater();
            _mavlinkDecoder = NULL;
        }
    
        // This needs to happen before we get into the QWidget dtor
        // otherwise  the QML engine reads freed data and tries to
        // destroy MainWindow a second time.
        delete _mainQmlWidgetHolder;
    
    Don Gagne's avatar
    Don Gagne committed
        _instance = NULL;
    
    dogmaphobic's avatar
    dogmaphobic committed
    QString MainWindow::_getWindowGeometryKey()
    
    #ifndef __mobile__
    
    MAVLinkDecoder* MainWindow::_mavLinkDecoderInstance(void)
    
        if (_mavlinkDecoder) {
            _mavlinkDecoder = new MAVLinkDecoder(qgcApp()->toolbox()->mavlinkProtocol());
            connect(_mavlinkDecoder, &MAVLinkDecoder::valueChanged, this, &MainWindow::valueChanged);
        }
    
        return _mavlinkDecoder;
    }
    
    void MainWindow::_buildCommonWidgets(void)
    {
    
    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);
    
    Don Gagne's avatar
    Don Gagne committed
        // Populate widget menu
    
        for (int i = 0, end = ARRAY_SIZE(rgDockWidgetNames); i < end; i++) {
    
    Don Gagne's avatar
    Don Gagne committed
            if (i == ONBOARD_FILES) {
                // Temporarily removed until twe can fix all the problems with it
                continue;
            }
    
            const char* pDockWidgetName = rgDockWidgetNames[i];
    
            // Add to menu
    
    Don Gagne's avatar
    Don Gagne committed
            QAction* action = new QAction(pDockWidgetName, this);
    
            action->setCheckable(true);
    
            action->setData(i);
    
            connect(action, &QAction::triggered, this, &MainWindow::_showDockWidgetAction);
            _ui.menuWidgets->addAction(action);
            _mapName2Action[pDockWidgetName] = action;
    
    /// Shows or hides the specified dock widget, creating if necessary
    void MainWindow::_showDockWidget(const QString& name, bool show)
    
    Don Gagne's avatar
    Don Gagne committed
        if (name == rgDockWidgetNames[ONBOARD_FILES]) {
            // Temporarily disabled due to bugs
            return;
        }
    
    
        // Create the inner widget if we need to
    
        if (!_mapName2DockWidget.contains(name)) {
    
    dogmaphobic's avatar
    dogmaphobic committed
            if(!_createInnerDockWidget(name)) {
    
    DonLakeFlyer's avatar
    DonLakeFlyer committed
                qWarning() << "Trying to load non existent widget:" << name;
    
    dogmaphobic's avatar
    dogmaphobic committed
                return;
            }
    
        Q_ASSERT(_mapName2DockWidget.contains(name));
    
        QGCDockWidget* dockWidget = _mapName2DockWidget[name];
    
        Q_ASSERT(dockWidget);
        dockWidget->setVisible(show);
    
        Q_ASSERT(_mapName2Action.contains(name));
        _mapName2Action[name]->setChecked(show);
    
    }
    
    /// Creates the specified inner dock widget and adds to the QDockWidget
    
    dogmaphobic's avatar
    dogmaphobic committed
    bool MainWindow::_createInnerDockWidget(const QString& widgetName)
    
        QGCDockWidget* widget = NULL;
    
        QAction *action = _mapName2Action[widgetName];
    
    dogmaphobic's avatar
    dogmaphobic committed
        if(action) {
            switch(action->data().toInt()) {
                case MAVLINK_INSPECTOR:
                    widget = new QGCMAVLinkInspector(widgetName, action, qgcApp()->toolbox()->mavlinkProtocol(),this);
                    break;
                case CUSTOM_COMMAND:
                    widget = new CustomCommandWidget(widgetName, action, this);
                    break;
                case ONBOARD_FILES:
                    widget = new QGCUASFileViewMulti(widgetName, action, this);
                    break;
                case HIL_CONFIG:
                    widget = new HILDockWidget(widgetName, action, this);
                    break;
                case ANALYZE:
    
                    widget = new Linecharts(widgetName, action, _mavLinkDecoderInstance(), this);
    
    dogmaphobic's avatar
    dogmaphobic committed
                    break;
            }
            if(widget) {
                _mapName2DockWidget[widgetName] = widget;
            }
    
    dogmaphobic's avatar
    dogmaphobic committed
        return widget != NULL;
    
    void MainWindow::_hideAllDockWidgets(void)
    {
    
        foreach(QGCDockWidget* dockWidget, _mapName2DockWidget) {
    
            dockWidget->setVisible(false);
        }
    }
    
    void MainWindow::_showDockWidgetAction(bool show)
    {
    
        QAction* action = qobject_cast<QAction*>(QObject::sender());
    
        Q_ASSERT(action);
    
        _showDockWidget(rgDockWidgetNames[action->data().toInt()], show);
    
    void MainWindow::showStatusBarCallback(bool checked)
    {
        _showStatusBar = checked;
        checked ? statusBar()->show() : statusBar()->hide();
    }
    
    
    void MainWindow::reallyClose(void)
    
        _forceClose = true;
        close();
    
    void MainWindow::closeEvent(QCloseEvent *event)
    {
    
        if (!_forceClose) {
    
    Ricardo de Almeida Gonzaga's avatar
    Ricardo de Almeida Gonzaga committed
            // Attempt close from within the root Qml item
    
            qgcApp()->qmlAttemptWindowClose();
    
            event->ignore();
            return;
    
        }
    
        // Should not be any active connections
    
        if (qgcApp()->toolbox()->multiVehicleManager()->activeVehicle()) {
    
    Don Gagne's avatar
    Don Gagne committed
            qWarning() << "All links should be disconnected by now";
        }
    
        _storeCurrentViewState();
    
    
        emit mainWindowClosed();
    
    }
    
    void MainWindow::loadSettings()
    {
    
        settings.beginGroup(MAIN_SETTINGS_GROUP);
    
        _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("LOW_POWER_MODE",     _lowPowerMode);
        settings.setValue("SHOW_STATUSBAR",     _showStatusBar);
    
        settings.endGroup();
    
    dogmaphobic's avatar
    dogmaphobic committed
        settings.setValue(_getWindowGeometryKey(), saveGeometry());
    
    #ifndef __mobile__
        _storeVisibleWidgetsSettings();
    #endif
    
    }
    
    void MainWindow::configureWindowName()
    {
    
        setWindowTitle(qApp->applicationName() + " " + qApp->applicationVersion());
    
    }
    
    /**
    * @brief Create all actions associated to the main window
    *
    **/
    void MainWindow::connectCommonActions()
    {
    
        // Connect internal actions
    
        connect(qgcApp()->toolbox()->multiVehicleManager(), &MultiVehicleManager::vehicleAdded, this, &MainWindow::_vehicleAdded);
    
    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))) {
    
            qgcApp()->showMessage(QString("Could not open information in browser: %1").arg(errorMessage));
    
    void MainWindow::_vehicleAdded(Vehicle* vehicle)
    
        connect(vehicle->uas(), &UAS::valueChanged, this, &MainWindow::valueChanged);
    
    /// Stores the state of the toolbar, status bar and widgets associated with the current view
    void MainWindow::_storeCurrentViewState(void)
    
    Don Gagne's avatar
    Don Gagne committed
    #ifndef __mobile__
    
        foreach(QGCDockWidget* dockWidget, _mapName2DockWidget) {
            dockWidget->saveSettings();
    
    Don Gagne's avatar
    Don Gagne committed
    #endif
    
    dogmaphobic's avatar
    dogmaphobic committed
        settings.setValue(_getWindowGeometryKey(), saveGeometry());
    
    /// @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);
    }
    
    
    #ifdef QGC_MOUSE_ENABLED_LINUX
    
    bool MainWindow::x11Event(XEvent *event)
    {
        emit x11EventOccured(event);
    
    #endif // QGC_MOUSE_ENABLED_LINUX
    
    
    #ifdef UNITTEST_BUILD
    void MainWindow::_showQmlTestWidget(void)
    {
        new QmlTestWidget();
    }
    #endif
    
    
    #ifndef __mobile__
    void MainWindow::_loadVisibleWidgetsSettings(void)
    {
        QSettings settings;
    
        QString widgets = settings.value(_visibleWidgetsKey).toString();
    
        if (!widgets.isEmpty()) {
            QStringList nameList = widgets.split(",");
    
            foreach (const QString &name, nameList) {
    
                _showDockWidget(name, true);
            }
        }
    }
    
    void MainWindow::_storeVisibleWidgetsSettings(void)
    {
        QString widgetNames;
        bool firstWidget = true;
    
        foreach (const QString &name, _mapName2DockWidget.keys()) {
    
            if (_mapName2DockWidget[name]->isVisible()) {
                if (!firstWidget) {
                    widgetNames += ",";
                } else {
                    firstWidget = false;
                }
    
                widgetNames += name;
            }
        }
    
        QSettings settings;
    
        settings.setValue(_visibleWidgetsKey, widgetNames);
    }
    #endif
    
    Don Gagne's avatar
    Don Gagne committed
    QObject* MainWindow::rootQmlObject(void)
    
    Don Gagne's avatar
    Don Gagne committed
        return _mainQmlWidgetHolder->getRootObject();
    
    
    void MainWindow::_showAdvancedUIChanged(bool advanced)
    {
        if (advanced) {
            menuBar()->addMenu(_ui.menuFile);
            menuBar()->addMenu(_ui.menuWidgets);
        } else {
            menuBar()->clear();
        }
    }