Skip to content
MainToolBar.qml 16.4 KiB
Newer Older
dogmaphobic's avatar
dogmaphobic committed
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009, 2015 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 QGC Main Tool Bar
 *   @author Gus Grubba <mavlink@grubba.com>
 */

import QtQuick 2.3
import QtQuick.Controls 1.2
import QtQuick.Controls.Styles 1.2

import QGroundControl.Controls              1.0
import QGroundControl.FactControls          1.0
import QGroundControl.Palette               1.0
import QGroundControl.MultiVehicleManager   1.0
import QGroundControl.ScreenTools           1.0
import QGroundControl.Controllers           1.0
Item {
    id:     toolBarHolder
    height: toolBarHeight
Don Gagne's avatar
Don Gagne committed
    QGCPalette { id: qgcPal; colorGroupEnabled: true }
    property var activeVehicle: multiVehicleManager.activeVehicle

Don Gagne's avatar
Don Gagne committed
    readonly property real toolBarHeight:   ScreenTools.defaultFontPixelHeight * 3
    property int cellSpacerSize:            ScreenTools.isMobile ? getProportionalDimmension(6) : getProportionalDimmension(4)
Don Gagne's avatar
Don Gagne committed
    readonly property int cellHeight:       toolBarHeight * 0.75

    readonly property real horizontalMargins:   ScreenTools.defaultFontPixelWidth / 2
    readonly property real verticalMargins:     ScreenTools.defaultFontPixelHeight / 4
Don Gagne's avatar
Don Gagne committed
    readonly property var colorBlue:    "#1a6eaa"
    readonly property var colorGreen:   "#329147"
    readonly property var colorRed:     "#942324"
    readonly property var colorOrange:  "#a76f26"
    readonly property var colorWhite:   "#f0f0f0"
    property var colorOrangeText: (qgcPal.globalTheme === QGCPalette.Light) ? "#b75711" : "#ea8225"
    property var colorRedText:    (qgcPal.globalTheme === QGCPalette.Light) ? "#ee1112" : "#ef2526"
    property var colorGreenText:  (qgcPal.globalTheme === QGCPalette.Light) ? "#046b1b" : "#00d930"
    property var colorWhiteText:  (qgcPal.globalTheme === QGCPalette.Light) ? "#343333" : "#f0f0f0"

    MainToolBarController { id: _controller }
    function showToolbarMessage(message) {
        toolBarMessage.text = message
        if (toolBarMessage.contentHeight > toolBarMessageCloseButton.height) {
Don Gagne's avatar
Don Gagne committed
            toolBarHolder.height = toolBarHeight + toolBarMessage.contentHeight + (verticalMargins * 2)
Don Gagne's avatar
Don Gagne committed
            toolBarHolder.height = toolBarHeight + toolBarMessageCloseButton.height + (verticalMargins * 2)
Don Gagne's avatar
Don Gagne committed
        }
        toolBarMessageArea.visible = true
Don Gagne's avatar
Don Gagne committed
    }

dogmaphobic's avatar
dogmaphobic committed
    function getProportionalDimmension(val) {
Don Gagne's avatar
Don Gagne committed
        return toolBarHeight * val / 40
dogmaphobic's avatar
dogmaphobic committed
    function getMessageColor() {
        if (activeVehicle.messageTypeNone)
dogmaphobic's avatar
dogmaphobic committed
            return qgcPal.button;
        if (activeVehicle.messageTypeNorma)
dogmaphobic's avatar
dogmaphobic committed
            return colorBlue;
        if (activeVehicle.messageTypeWarning)
dogmaphobic's avatar
dogmaphobic committed
            return colorOrange;
        if (activeVehicle.messageTypeError)
dogmaphobic's avatar
dogmaphobic committed
            return colorRed;
        // Cannot be so make make it obnoxious to show error
        return "purple";
    }

    function getMessageIcon() {
        if (activeVehicle.messageTypeNormal || activeVehicle.messageTypeNone)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Megaphone";
dogmaphobic's avatar
dogmaphobic committed
        else
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Yield";
dogmaphobic's avatar
dogmaphobic committed
    }

    function getBatteryIcon() {
        if(activeVehicle.batteryPercent < 20.0)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_0";
        else if(activeVehicle.batteryPercent < 40.0)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_20";
        else if(activeVehicle.batteryPercent < 60.0)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_40";
        else if(activeVehicle.batteryPercent < 80.0)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_60";
        else if(activeVehicle.batteryPercent < 90.0)
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_80";
dogmaphobic's avatar
dogmaphobic committed
        else
Don Gagne's avatar
Don Gagne committed
            return "qrc:/res/Battery_100";
dogmaphobic's avatar
dogmaphobic committed
    function getBatteryColor() {
        if (activeVehicle.batteryPercent > 40.0)
dogmaphobic's avatar
dogmaphobic committed
            return colorGreen;
        if(activeVehicle.batteryPercent > 0.01)
dogmaphobic's avatar
dogmaphobic committed
            return colorRed;
        // This means there is no battery level data
        return colorBlue;
    }

    function getSatelliteColor() {
        // No GPS data
        if (activeVehicle.satelliteCount < 0)
dogmaphobic's avatar
dogmaphobic committed
            return qgcPal.button
        // No Lock
        if(activeVehicle.satelliteLock < 2)
dogmaphobic's avatar
dogmaphobic committed
            return colorRed;
        // 2D Lock
        if(activeVehicle.satelliteLock === 2)
dogmaphobic's avatar
dogmaphobic committed
            return colorBlue;
        // Lock is 3D or more
        return colorGreen;
    }

    function getRSSIColor(value) {
        if(value < 10)
            return colorRed;
        if(value < 50)
            return colorOrange;
        return colorGreen;
dogmaphobic's avatar
dogmaphobic committed
    function showMavStatus() {
         return (multiVehicleManager.activeVehicleAvailable && activeVehicle.heartbeatTimeout === 0 && _controller.connectionCount > 0);
dogmaphobic's avatar
dogmaphobic committed
    //-------------------------------------------------------------------------
    //-- Main menu for Mobile Devices
    Menu {
Don Gagne's avatar
Don Gagne committed
        id: mobileMenu
dogmaphobic's avatar
dogmaphobic committed
        ExclusiveGroup { id: mainMenuGroup }
dogmaphobic's avatar
dogmaphobic committed
        MenuItem {
            id:             flyViewShowing
            text:           "Fly"
            checkable:      true
            checked:        true
dogmaphobic's avatar
dogmaphobic committed
            exclusiveGroup: mainMenuGroup

            onTriggered: {
                checked = true
                _controller.onFlyView();
dogmaphobic's avatar
dogmaphobic committed
            }
        }
dogmaphobic's avatar
dogmaphobic committed
        MenuItem {
            id:             setupViewShowing
            text:           "Setup"
            checkable:      true
dogmaphobic's avatar
dogmaphobic committed
            exclusiveGroup: mainMenuGroup

            onTriggered: {
                checked = true
                _controller.onSetupView();
dogmaphobic's avatar
dogmaphobic committed
            }
        }
dogmaphobic's avatar
dogmaphobic committed
        MenuItem {
            id:             planViewShowing
            text:           "Plan"
            checkable:      true
dogmaphobic's avatar
dogmaphobic committed
            exclusiveGroup: mainMenuGroup

            onTriggered: {
                checked = true
                _controller.onPlanView();
dogmaphobic's avatar
dogmaphobic committed
            }
        }
Don Gagne's avatar
Don Gagne committed

        MenuSeparator { }


        MenuItem {
            text:           "QGroundControl Settings"

            onTriggered: controller.showSettings()
        }
Don Gagne's avatar
Don Gagne committed
    } // Menu
dogmaphobic's avatar
dogmaphobic committed

Don Gagne's avatar
Don Gagne committed
    /*
    Row {
        id:         toolRow
        height:     cellHeight
        spacing:    getProportionalDimmension(4)
Don Gagne's avatar
Don Gagne committed
    }
    */

    Loader {
        id:                 desktopToolsLoader
        height:             cellHeight
        x:                  horizontalMargins
        y:                  (toolBarHeight - cellHeight) / 2
        sourceComponent:    ScreenTools.isMobile ? undefined : desktopTools
    }
Don Gagne's avatar
Don Gagne committed
    //---------------------------------------------------------------------
    //-- Indicators
    Row {
        id:                     row12
        x:                      horizontalMargins + (ScreenTools.isMobile ? 0: desktopToolsLoader.item.width)
Don Gagne's avatar
Don Gagne committed
        height:                 cellHeight
        spacing:                cellSpacerSize
        anchors.top:            desktopToolsLoader.top
        anchors.verticalCenter: desktopToolsLoader.verticalCenter
Don Gagne's avatar
Don Gagne committed
        //-- "Hamburger" menu for Mobile Devices
        Item {
            id:         actionButton
            visible:    ScreenTools.isMobile
            height:     cellHeight
            width:      cellHeight
            Image {
                id:             buttomImg
                anchors.fill:   parent
                source:         "/qmlimages/buttonMore.svg"
                mipmap:         true
                smooth:         true
                antialiasing:   true
                fillMode:       Image.PreserveAspectFit
Don Gagne's avatar
Don Gagne committed
            MouseArea {
                anchors.fill: parent
                acceptedButtons: Qt.LeftButton
Don Gagne's avatar
Don Gagne committed
                    if (mouse.button == Qt.LeftButton)
                    {
                        mobileMenu.popup();
Don Gagne's avatar
Don Gagne committed
        }
Don Gagne's avatar
Don Gagne committed
        //-- Separator if Hamburger menu is visible
        Rectangle {
            visible:    actionButton.visible
            height:     cellHeight
            width:      cellHeight
            color:      "#00000000"
            anchors.verticalCenter: parent.verticalCenter
        }
Don Gagne's avatar
Don Gagne committed
        Loader {
            id:         activeVehicleLoader
            visible:    showMavStatus()
            source:     multiVehicleManager.activeVehicleAvailable ? "MainToolBarActiveVehicleComponent.qml" : ""
Don Gagne's avatar
Don Gagne committed
            property real cellHeight:       toolBarHolder.cellHeight
            property real cellSpacerSize:   toolBarHolder.cellSpacerSize
        }
Don Gagne's avatar
Don Gagne committed
        Rectangle {
            id: connectionStatus
            width: getProportionalDimmension(160)
            height: cellHeight
            visible: (_controller.connectionCount > 0 && multiVehicleManager.activeVehicleAvailable && activeVehicle.heartbeatTimeout != 0)
            anchors.verticalCenter: parent.verticalCenter
            color: "#00000000"
            border.color: "#00000000"
            border.width: 0

            QGCLabel {
                id: connectionStatusText
                text: qsTr("CONNECTION LOST")
                font.pixelSize: ScreenTools.defaultFontPixelSize
                font.weight: Font.DemiBold
                anchors.verticalCenter: parent.verticalCenter
Don Gagne's avatar
Don Gagne committed
                anchors.horizontalCenter: parent.horizontalCenter
                color: colorRedText
dogmaphobic's avatar
dogmaphobic committed
            }
Don Gagne's avatar
Don Gagne committed
        }
Don Gagne's avatar
Don Gagne committed
    } // Row
dogmaphobic's avatar
dogmaphobic committed

    Row {
Don Gagne's avatar
Don Gagne committed
        id:                     connectRow
        anchors.rightMargin:    verticalMargins
        anchors.right:          parent.right
Don Gagne's avatar
Don Gagne committed
        anchors.top:            desktopToolsLoader.top
        anchors.verticalCenter: desktopToolsLoader.verticalCenter
        height:                 desktopToolsLoader.height
Don Gagne's avatar
Don Gagne committed
        spacing:                cellSpacerSize
dogmaphobic's avatar
dogmaphobic committed
        Menu {
            id: connectMenu
dogmaphobic's avatar
dogmaphobic committed
            Component.onCompleted: {
                _controller.configListChanged.connect(connectMenu.updateConnectionList);
dogmaphobic's avatar
dogmaphobic committed
                connectMenu.updateConnectionList();
dogmaphobic's avatar
dogmaphobic committed
            }
dogmaphobic's avatar
dogmaphobic committed
            function addMenuEntry(name) {
                var label = "Add Connection"
                if(name !== "")
                    label = name;
                var mItem = connectMenu.addItem(label);
                var menuSlot = function() {_controller.onConnect(name)};
dogmaphobic's avatar
dogmaphobic committed
                mItem.triggered.connect(menuSlot);
            }
            function updateConnectionList() {
                connectMenu.clear();
                for(var i = 0; i < _controller.configList.length; i++) {
                    connectMenu.addMenuEntry(_controller.configList[i]);
dogmaphobic's avatar
dogmaphobic committed
                }
                if(_controller.configList.length > 0) {
dogmaphobic's avatar
dogmaphobic committed
                    connectMenu.addSeparator();
                }
                // Add "Add Connection" to the list
                connectMenu.addMenuEntry("");
dogmaphobic's avatar
dogmaphobic committed
            }
        }

        QGCButton {
dogmaphobic's avatar
dogmaphobic committed
            id:         connectButton
dogmaphobic's avatar
dogmaphobic committed
            width:      getProportionalDimmension(100)
            visible:    _controller.connectionCount === 0
dogmaphobic's avatar
dogmaphobic committed
            text:       qsTr("Connect")
            menu:       connectMenu
        }

        QGCButton {
            id:         disconnectButton
dogmaphobic's avatar
dogmaphobic committed
            width:      getProportionalDimmension(100)
            visible:    _controller.connectionCount === 1
dogmaphobic's avatar
dogmaphobic committed
            text:       qsTr("Disconnect")
dogmaphobic's avatar
dogmaphobic committed
            onClicked: {
                _controller.onDisconnect("");
dogmaphobic's avatar
dogmaphobic committed
            }
        }

        Menu {
            id: disconnectMenu
            Component.onCompleted: {
                _controller.connectedListChanged.connect(disconnectMenu.onConnectedListChanged)
dogmaphobic's avatar
dogmaphobic committed
            }
dogmaphobic's avatar
dogmaphobic committed
            function addMenuEntry(name) {
                var mItem = disconnectMenu.addItem(name);
                var menuSlot = function() {_controller.onDisconnect(name)};
dogmaphobic's avatar
dogmaphobic committed
                mItem.triggered.connect(menuSlot);
            }
dogmaphobic's avatar
dogmaphobic committed
            function onConnectedListChanged(conList) {
                disconnectMenu.clear();
                for(var i = 0; i < conList.length; i++) {
dogmaphobic's avatar
dogmaphobic committed
                    disconnectMenu.addMenuEntry(conList[i]);
dogmaphobic's avatar
dogmaphobic committed
            id:         multidisconnectButton
dogmaphobic's avatar
dogmaphobic committed
            width:      getProportionalDimmension(100)
dogmaphobic's avatar
dogmaphobic committed
            text:       "Disconnect"
            visible:    _controller.connectionCount > 1
dogmaphobic's avatar
dogmaphobic committed
            menu:       disconnectMenu
dogmaphobic's avatar
dogmaphobic committed
        }
Don Gagne's avatar
Don Gagne committed
    } // Row

    // Progress bar
    Rectangle {
Don Gagne's avatar
Don Gagne committed
        id:             progressBar
Don Gagne's avatar
Don Gagne committed
        anchors.top:    desktopToolsLoader.bottom
Don Gagne's avatar
Don Gagne committed
        height:         getProportionalDimmension(3)
        width:          parent.width * _controller.progressBarValue
Don Gagne's avatar
Don Gagne committed
        color:          qgcPal.text
Don Gagne's avatar
Don Gagne committed
    // Toolbar message area
    Rectangle {
        id:                     toolBarMessageArea
        anchors.leftMargin:     horizontalMargins
        anchors.rightMargin:    horizontalMargins
        anchors.topMargin:      verticalMargins
        anchors.bottomMargin:   verticalMargins
        anchors.top:            progressBar.bottom
        anchors.bottom:         parent.bottom
        anchors.left:           parent.left
        anchors.right:          parent.right
        color:                  qgcPal.windowShadeDark
        visible:                false
Don Gagne's avatar
Don Gagne committed

        QGCLabel {
            id:             toolBarMessage
            anchors.fill:   parent
            wrapMode:       Text.WordWrap
			color:			qgcPal.warningText
Don Gagne's avatar
Don Gagne committed
        }

        QGCButton {
            id:                     toolBarMessageCloseButton
            anchors.rightMargin:    horizontalMargins
            anchors.top:            parent.top
            anchors.right:          parent.right
			primary:				true
Don Gagne's avatar
Don Gagne committed
            text:                   "Close Message"

            onClicked: {
                parent.visible = false
Don Gagne's avatar
Don Gagne committed
                toolBarHolder.height = toolBarHeight
                _controller.onToolBarMessageClosed()
Don Gagne's avatar
Don Gagne committed

    Component {
        id: desktopTools

        //---------------------------------------------------------------------
        //-- Main menu for Non Mobile Devices (Chevron Buttons)
        Row {
            id:             row11
            height:         cellHeight
            spacing:        -getProportionalDimmension(12)

            Connections {
                target: ScreenTools
                onRepaintRequested: {
                    setupButton.repaintChevron   = true;
                    planButton.repaintChevron    = true;
                    flyButton.repaintChevron     = true;
                }
            }

            ExclusiveGroup { id: mainActionGroup }

            QGCToolBarButton {
                id:             setupButton
                width:          getProportionalDimmension(90)
                height:         cellHeight
                exclusiveGroup: mainActionGroup
                text:           "Setup"

                onClicked: {
                    checked = true
                    _controller.onSetupView();
                }
                z: 1000
            }

            QGCToolBarButton {
                id:             planButton
                width:          getProportionalDimmension(90)
                height:         cellHeight
                exclusiveGroup: mainActionGroup
                text:           "Plan"

                onClicked: {
                    checked = true
                    _controller.onPlanView();
                }
                z: 900
            }

            QGCToolBarButton {
                id:             flyButton
                width:          getProportionalDimmension(90)
                height:         cellHeight
                exclusiveGroup: mainActionGroup
                text:           "Fly"
                checked:        true

                onClicked: {
                    checked = true
                    _controller.onFlyView();
                }
                z: 800
            }
        } // Row
    } // Component - desktopTools
Don Gagne's avatar
Don Gagne committed
} // Rectangle