diff --git a/qgroundcontrol.qrc b/qgroundcontrol.qrc
index dd64f5fdfbe..690d3692ce6 100644
--- a/qgroundcontrol.qrc
+++ b/qgroundcontrol.qrc
@@ -127,6 +127,9 @@
src/QmlControls/MissionItemIndexLabel.qml
src/PlanView/MissionItemMapVisual.qml
src/PlanView/MissionItemStatus.qml
+ src/QmlControls/ModeSwitchDisplay.qml
+ src/QmlControls/MultiRotorMotorDisplay.qml
+ src/QmlControls/MvPanelPage.qml
src/QmlControls/OfflineMapButton.qml
src/QtLocationPlugin/QMLControl/OfflineMapEditor.qml
src/UI/preferences/OfflineMapInfo.qml
@@ -161,12 +164,14 @@
src/QmlControls/QGCMenuSeparator.qml
src/QmlControls/QGCMouseArea.qml
src/QmlControls/QGCMovableItem.qml
+ src/QmlControls/QGCPageIndicator.qml
src/QmlControls/QGCPopupDialog.qml
src/QmlControls/PipView.qml
src/QmlControls/PipState.qml
src/QmlControls/QGCRadioButton.qml
src/QmlControls/QGCSimpleMessageDialog.qml
src/QmlControls/QGCSlider.qml
+ src/QmlControls/QGCSwipeView.qml
src/QmlControls/QGCSwitch.qml
src/QmlControls/QGCTabBar.qml
src/QmlControls/QGCTabButton.qml
@@ -342,6 +347,7 @@
src/Viewer3D/Viewer3DQml/Models3D/Line3D.qml
src/Viewer3D/Viewer3DQml/Models3D/Viewer3DVehicleItems.qml
src/Viewer3D/Viewer3DQml/Viewer3DProgressBar.qml
+ src/FlightDisplay/FlyViewTopRightPanel.qml
src/FirstRunPromptDialogs/UnitsFirstRunPrompt.qml
diff --git a/src/FlightDisplay/FlyViewTopRightColumnLayout.qml b/src/FlightDisplay/FlyViewTopRightColumnLayout.qml
index 9a7b8197215..332ee4af10b 100644
--- a/src/FlightDisplay/FlyViewTopRightColumnLayout.qml
+++ b/src/FlightDisplay/FlyViewTopRightColumnLayout.qml
@@ -20,28 +20,7 @@ import QGroundControl.ScreenTools
ColumnLayout {
width: _rightPanelWidth
- RowLayout {
- id: multiVehiclePanelSelector
- Layout.alignment: Qt.AlignTop
- spacing: ScreenTools.defaultFontPixelWidth
- visible: QGroundControl.multiVehicleManager.vehicles.count > 1 && QGroundControl.corePlugin.options.flyView.showMultiVehicleList
-
- QGCMapPalette { id: mapPal; lightColors: true }
-
- QGCRadioButton {
- id: singleVehicleRadio
- text: qsTr("Single")
- checked: _showSingleVehicleUI
- onClicked: _showSingleVehicleUI = true
- textColor: mapPal.text
- }
-
- QGCRadioButton {
- text: qsTr("Multi-Vehicle")
- textColor: mapPal.text
- onClicked: _showSingleVehicleUI = false
- }
- }
+ property bool mvPanelVisible: false
TerrainProgress {
Layout.alignment: Qt.AlignTop
@@ -54,7 +33,7 @@ ColumnLayout {
Loader {
id: photoVideoControlLoader
Layout.alignment: Qt.AlignTop | Qt.AlignRight
- sourceComponent: globals.activeVehicle && _showSingleVehicleUI ? photoVideoControlComponent : undefined
+ sourceComponent: globals.activeVehicle && !mvPanelVisible ? photoVideoControlComponent : undefined
property real rightEdgeCenterInset: visible ? parent.width - x : 0
@@ -65,10 +44,4 @@ ColumnLayout {
}
}
}
-
- MultiVehicleList {
- Layout.preferredWidth: _rightPanelWidth
- Layout.fillHeight: true
- visible: !_showSingleVehicleUI
- }
}
diff --git a/src/FlightDisplay/FlyViewTopRightPanel.qml b/src/FlightDisplay/FlyViewTopRightPanel.qml
new file mode 100644
index 00000000000..af4a00aaf90
--- /dev/null
+++ b/src/FlightDisplay/FlyViewTopRightPanel.qml
@@ -0,0 +1,213 @@
+/****************************************************************************
+ *
+ * (c) 2009-2020 QGROUNDCONTROL PROJECT
+ *
+ * QGroundControl is licensed according to the terms in the file
+ * COPYING.md in the root of the source code directory.
+ *
+ ****************************************************************************/
+
+import QtQuick
+import QtQuick.Layouts
+
+import QGroundControl
+import QGroundControl.Controls
+import QGroundControl.FlightDisplay
+import QGroundControl.FlightMap
+import QGroundControl.Palette
+import QGroundControl.ScreenTools
+
+
+Item {
+
+ property bool panelVisible: togglePanelBtn.checked
+ property alias toggleBtn: togglePanelBtn
+
+ Rectangle {
+ id: topRightPanel
+ anchors.fill: parent
+ color: qgcPal.toolbarBackground
+ visible: !QGroundControl.videoManager.fullScreen && togglePanelBtn.checked
+ clip: true
+
+ QGCPalette { id: qgcPal }
+
+ MultiVehicleList {
+ id: multiVehicleList
+ anchors.top: parent.top
+ anchors.bottom: parent.verticalCenter
+ anchors.right: parent.right
+ anchors.left: parent.left
+ anchors.margins: ScreenTools.defaultFontPixelHeight / 2
+
+ Rectangle {
+ anchors.fill: parent
+
+ gradient: Gradient {
+ orientation: Gradient.Vertical
+
+ GradientStop { position: 0.95; color: "transparent" }
+ GradientStop { position: 1.0; color: topRightPanel.color }
+ }
+
+ Rectangle {
+ anchors.left: parent.left
+ anchors.right: parent.right
+ anchors.bottom: parent.bottom
+ height: 1
+ color: QGroundControl.globalPalette.groupBorder
+ }
+ }
+
+ }
+
+ QGCSwipeView {
+ id: swipePages
+ anchors.top: parent.verticalCenter
+ anchors.bottom: parent.bottom
+ anchors.right: parent.right
+ anchors.left: parent.left
+
+ MvPanelPage {
+ id: buttonsPage
+
+ ColumnLayout {
+ anchors.right: parent.right
+ anchors.left: parent.left
+ anchors.verticalCenter: parent.verticalCenter
+ anchors.bottomMargin: ScreenTools.defaultFontPixelHeight * 3
+ spacing: ScreenTools.defaultFontPixelHeight / 2
+
+ QGCLabel {
+ text: qsTr("Multi Vehicle Selection")
+ Layout.alignment: Qt.AlignHCenter
+ }
+
+ RowLayout {
+ id: selectionRowLayout
+ Layout.alignment: Qt.AlignHCenter
+
+ QGCButton {
+ text: qsTr("Select All")
+ enabled: multiVehicleList.selectedVehicles && multiVehicleList.selectedVehicles.count !== QGroundControl.multiVehicleManager.vehicles.count
+ onClicked: multiVehicleList.selectAll()
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 5
+ }
+
+ QGCButton {
+ text: qsTr("Deselect All")
+ enabled: multiVehicleList.selectedVehicles && multiVehicleList.selectedVehicles.count > 0
+ onClicked: multiVehicleList.deselectAll()
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 5
+
+ }
+ }
+
+
+ QGCLabel {
+ text: qsTr("Multi Vehicle Actions")
+ Layout.alignment: Qt.AlignHCenter
+ }
+
+ RowLayout {
+ id: actionRowLayout
+ Layout.alignment: Qt.AlignHCenter
+
+ QGCButton {
+ text: qsTr("Arm")
+ enabled: multiVehicleList.armAvailable()
+ onClicked: _guidedController.confirmAction(_guidedController.actionMVArm)
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 2.75
+ leftPadding: 0
+ rightPadding: 0
+ }
+
+ QGCButton {
+ text: qsTr("Disarm")
+ enabled: multiVehicleList.disarmAvailable()
+ onClicked: _guidedController.confirmAction(_guidedController.actionMVDisarm)
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 2.75
+ leftPadding: 0
+ rightPadding: 0
+ }
+
+ QGCButton {
+ text: qsTr("Start")
+ enabled: multiVehicleList.startAvailable()
+ onClicked: _guidedController.confirmAction(_guidedController.actionMVStartMission)
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 2.75
+ leftPadding: 0
+ rightPadding: 0
+ }
+
+ QGCButton {
+ text: qsTr("Pause")
+ enabled: multiVehicleList.pauseAvailable()
+ onClicked: _guidedController.confirmAction(_guidedController.actionMVPause)
+ Layout.preferredWidth: ScreenTools.defaultFontPixelHeight * 2.75
+ leftPadding: 0
+ rightPadding: 0
+ }
+ }
+ }
+ } // Page 1
+
+ MvPanelPage {
+
+ // We use a Loader to load the photoVideoControlComponent only when the active vehicle is not null
+ // This make it easier to implement PhotoVideoControl without having to check for the mavlink camera
+ // to be null all over the place
+
+ Loader {
+ id: photoVideoControlLoader
+ anchors.horizontalCenter: parent.horizontalCenter
+ anchors.bottomMargin: ScreenTools.defaultFontPixel
+ sourceComponent: globals.activeVehicle && togglePanelBtn.checked ? photoVideoControlComponent : undefined
+
+ property real rightEdgeCenterInset: visible ? parent.width - x : 0
+
+ Component {
+ id: photoVideoControlComponent
+
+ PhotoVideoControl {
+ }
+ }
+ }
+ } // Page 2
+ }
+
+ QGCPageIndicator {
+ id: pageIndicator
+ count: swipePages.count
+ currentIndex: swipePages.currentIndex
+ anchors.horizontalCenter: parent.horizontalCenter
+ anchors.bottom: parent.bottom
+ anchors.bottomMargin: togglePanelBtn.height
+
+ delegate: Rectangle {
+ height: ScreenTools.defaultFontPixelHeight / 2
+ width: height
+ radius: width / 2
+ color: model.index === pageIndicator.currentIndex ? qgcPal.text : qgcPal.button
+ opacity: model.index === pageIndicator.currentIndex ? 0.9 : 0.3
+ }
+ }
+ }
+
+ QGCButton {
+ id: togglePanelBtn
+ anchors.top: parent.top
+ anchors.horizontalCenter: parent.horizontalCenter
+ anchors.topMargin: topRightPanel.visible ? parent.height : 0
+ width: _rightPanelWidth / 5
+ height: _rightPanelWidth / 18
+ checkable: true
+
+ background: Rectangle {
+ radius: parent.height / 2
+ color: qgcPal.toolbarBackground
+ border.color: parent.checked ? QGroundControl.globalPalette.groupBorder : qgcPal.text
+ border.width: 1
+ }
+ }
+}
diff --git a/src/FlightDisplay/FlyViewWidgetLayer.qml b/src/FlightDisplay/FlyViewWidgetLayer.qml
index 870e8d7dca7..db8eb79048e 100644
--- a/src/FlightDisplay/FlyViewWidgetLayer.qml
+++ b/src/FlightDisplay/FlyViewWidgetLayer.qml
@@ -46,7 +46,7 @@ Item {
property real _margins: ScreenTools.defaultFontPixelWidth / 2
property real _toolsMargin: ScreenTools.defaultFontPixelWidth * 0.75
property rect _centerViewport: Qt.rect(0, 0, width, height)
- property real _rightPanelWidth: ScreenTools.defaultFontPixelWidth * 30
+ property real _rightPanelWidth: ScreenTools.defaultFontPixelWidth * 40
property alias _gripperMenu: gripperOptions
property real _layoutMargin: ScreenTools.defaultFontPixelWidth * 0.75
property bool _layoutSpacing: ScreenTools.defaultFontPixelWidth
@@ -59,24 +59,39 @@ Item {
leftEdgeTopInset: toolStrip.leftEdgeTopInset
leftEdgeCenterInset: toolStrip.leftEdgeCenterInset
leftEdgeBottomInset: virtualJoystickMultiTouch.visible ? virtualJoystickMultiTouch.leftEdgeBottomInset : parentToolInsets.leftEdgeBottomInset
- rightEdgeTopInset: topRightColumnLayout.rightEdgeTopInset
- rightEdgeCenterInset: topRightColumnLayout.rightEdgeCenterInset
+ rightEdgeTopInset: topRightPanel.rightEdgeTopInset
+ rightEdgeCenterInset: topRightPanel.rightEdgeCenterInset
rightEdgeBottomInset: bottomRightRowLayout.rightEdgeBottomInset
topEdgeLeftInset: toolStrip.topEdgeLeftInset
topEdgeCenterInset: mapScale.topEdgeCenterInset
- topEdgeRightInset: topRightColumnLayout.topEdgeRightInset
+ topEdgeRightInset: topRightPanel.topEdgeRightInset
bottomEdgeLeftInset: virtualJoystickMultiTouch.visible ? virtualJoystickMultiTouch.bottomEdgeLeftInset : parentToolInsets.bottomEdgeLeftInset
bottomEdgeCenterInset: bottomRightRowLayout.bottomEdgeCenterInset
bottomEdgeRightInset: virtualJoystickMultiTouch.visible ? virtualJoystickMultiTouch.bottomEdgeRightInset : bottomRightRowLayout.bottomEdgeRightInset
}
+ FlyViewTopRightPanel {
+ id: topRightPanel
+ anchors.top: parent.top
+ anchors.right: parent.right
+ anchors.rightMargin: _layoutMargin
+ width: _rightPanelWidth
+ height: _rightPanelWidth * 1.75
+
+ property real topEdgeRightInset: height + _layoutMargin
+ property real rightEdgeTopInset: width + _layoutMargin
+ property real rightEdgeCenterInset: rightEdgeTopInset
+ }
+
FlyViewTopRightColumnLayout {
id: topRightColumnLayout
anchors.margins: _layoutMargin
- anchors.top: parent.top
+ anchors.top: topRightPanel.top
anchors.bottom: bottomRightRowLayout.top
anchors.right: parent.right
+ anchors.topMargin: topRightPanel.toggleBtn.height + _toolsMargin
spacing: _layoutSpacing
+ mvPanelVisible: topRightPanel.panelVisible
property real topEdgeRightInset: childrenRect.height + _layoutMargin
property real rightEdgeTopInset: width + _layoutMargin
diff --git a/src/FlightDisplay/GuidedActionsController.qml b/src/FlightDisplay/GuidedActionsController.qml
index a8381956a6d..75df33d8b53 100644
--- a/src/FlightDisplay/GuidedActionsController.qml
+++ b/src/FlightDisplay/GuidedActionsController.qml
@@ -34,8 +34,10 @@ Item {
readonly property string emergencyStopTitle: qsTr("EMERGENCY STOP")
readonly property string armTitle: qsTr("Arm")
+ readonly property string mvArmTitle: qsTr("Arm (MV)")
readonly property string forceArmTitle: qsTr("Force Arm")
readonly property string disarmTitle: qsTr("Disarm")
+ readonly property string mvDisarmTitle: qsTr("Disarm (MV)")
readonly property string rtlTitle: qsTr("Return")
readonly property string takeoffTitle: qsTr("Takeoff")
readonly property string gripperTitle: qsTr("Gripper Function")
@@ -62,12 +64,15 @@ Item {
readonly property string changeHeadingTitle: qsTr("Change Heading")
readonly property string armMessage: qsTr("Arm the vehicle.")
+ readonly property string mvArmMessage: qsTr("Arm selected vehicles.")
readonly property string forceArmMessage: qsTr("WARNING: This will force arming of the vehicle bypassing any safety checks.")
readonly property string disarmMessage: qsTr("Disarm the vehicle")
+ readonly property string mvDisarmMessage: qsTr("Disarm selected vehicles.")
readonly property string emergencyStopMessage: qsTr("WARNING: THIS WILL STOP ALL MOTORS. IF VEHICLE IS CURRENTLY IN THE AIR IT WILL CRASH.")
readonly property string takeoffMessage: qsTr("Takeoff from ground and hold position.")
- readonly property string gripperMessage: qsTr("Grab or Release the cargo")
+ readonly property string gripperMessage: qsTr("Grab or Release the cargo")
readonly property string startMissionMessage: qsTr("Takeoff from ground and start the current mission.")
+ readonly property string mvStartMissionMessage: qsTr("Takeoff from ground and start the current mission for selected vehicles.")
readonly property string continueMissionMessage: qsTr("Continue the mission from the current waypoint.")
readonly property string resumeMissionUploadFailMessage: qsTr("Upload of resume mission failed. Confirm to retry upload")
readonly property string landMessage: qsTr("Land the vehicle at the current position.")
@@ -80,7 +85,7 @@ Item {
readonly property string orbitMessage: qsTr("Orbit the vehicle around the specified location.")
readonly property string landAbortMessage: qsTr("Abort the landing sequence.")
readonly property string pauseMessage: qsTr("Pause the vehicle at it's current position, adjusting altitude up or down as needed.")
- readonly property string mvPauseMessage: qsTr("Pause all vehicles at their current position.")
+ readonly property string mvPauseMessage: qsTr("Pause selected vehicles at their current position.")
readonly property string vtolTransitionFwdMessage: qsTr("Transition VTOL to fixed wing flight.")
readonly property string vtolTransitionMRMessage: qsTr("Transition VTOL to multi-rotor flight.")
readonly property string roiMessage: qsTr("Make the specified location a Region Of Interest.")
@@ -119,6 +124,10 @@ Item {
readonly property int actionSetEstimatorOrigin: 28
readonly property int actionSetFlightMode: 29
readonly property int actionChangeHeading: 30
+ readonly property int actionMVArm: 31
+ readonly property int actionMVDisarm: 32
+
+
readonly property int customActionStart: 10000 // Custom actions ids should start here so that they don't collide with the built in actions
@@ -417,6 +426,11 @@ Item {
confirmDialog.message = armMessage
confirmDialog.hideTrigger = Qt.binding(function() { return !showArm })
break;
+ case actionMVArm:
+ confirmDialog.title = mvArmTitle
+ confirmDialog.message = mvArmMessage
+ confirmDialog.hideTrigger = true
+ break;
case actionForceArm:
confirmDialog.title = forceArmTitle
confirmDialog.message = forceArmMessage
@@ -430,6 +444,11 @@ Item {
confirmDialog.message = disarmMessage
confirmDialog.hideTrigger = Qt.binding(function() { return !showDisarm })
break;
+ case actionMVDisarm:
+ confirmDialog.title = mvDisarmTitle
+ confirmDialog.message = mvDisarmMessage
+ confirmDialog.hideTrigger = true
+ break;
case actionEmergencyStop:
confirmDialog.title = emergencyStopTitle
confirmDialog.message = emergencyStopMessage
@@ -449,7 +468,7 @@ Item {
break;
case actionMVStartMission:
confirmDialog.title = mvStartMissionTitle
- confirmDialog.message = startMissionMessage
+ confirmDialog.message = mvStartMissionMessage
confirmDialog.hideTrigger = true
break;
case actionContinueMission:
@@ -576,7 +595,7 @@ Item {
// Executes the specified action
function executeAction(actionCode, actionData, sliderOutputValue, optionChecked) {
var i;
- var rgVehicle;
+ var selectedVehicles;
switch (actionCode) {
case actionRTL:
_activeVehicle.guidedModeRTL(optionChecked)
@@ -597,20 +616,35 @@ Item {
_activeVehicle.startMission()
break
case actionMVStartMission:
- rgVehicle = QGroundControl.multiVehicleManager.vehicles
- for (i = 0; i < rgVehicle.count; i++) {
- rgVehicle.get(i).startMission()
+ selectedVehicles = QGroundControl.multiVehicleManager.selectedVehicles
+ for (i = 0; i < selectedVehicles.count; i++) {
+ var vehicle = selectedVehicles.get(i)
+ if (vehicle.armed === true){
+ vehicle.startMission()
+ }
}
break
case actionArm:
_activeVehicle.armed = true
break
+ case actionMVArm:
+ selectedVehicles = QGroundControl.multiVehicleManager.selectedVehicles
+ for (i = 0; i < selectedVehicles.count; i++) {
+ selectedVehicles.get(i).armed = true
+ }
+ break
case actionForceArm:
_activeVehicle.forceArm()
break
case actionDisarm:
_activeVehicle.armed = false
break
+ case actionMVDisarm:
+ selectedVehicles = QGroundControl.multiVehicleManager.selectedVehicles
+ for (i = 0; i < selectedVehicles.count; i++) {
+ selectedVehicles.get(i).armed = false
+ }
+ break
case actionEmergencyStop:
_activeVehicle.emergencyStop()
break
@@ -638,9 +672,9 @@ Item {
_activeVehicle.guidedModeChangeAltitude(altitudeChangeInMeters, true /* pauseVehicle */)
break
case actionMVPause:
- rgVehicle = QGroundControl.multiVehicleManager.vehicles
- for (i = 0; i < rgVehicle.count; i++) {
- rgVehicle.get(i).pauseVehicle()
+ selectedVehicles = QGroundControl.multiVehicleManager.selectedVehicles
+ for (i = 0; i < selectedVehicles.count; i++) {
+ selectedVehicles.get(i).pauseVehicle()
}
break
case actionVtolTransitionToFwdFlight:
diff --git a/src/FlightDisplay/MultiVehicleList.qml b/src/FlightDisplay/MultiVehicleList.qml
index 73b2e56f7ae..310294942ca 100644
--- a/src/FlightDisplay/MultiVehicleList.qml
+++ b/src/FlightDisplay/MultiVehicleList.qml
@@ -19,66 +19,103 @@ import QGroundControl.Vehicle
import QGroundControl.FlightMap
Item {
- property real _margin: ScreenTools.defaultFontPixelWidth / 2
- property real _widgetHeight: ScreenTools.defaultFontPixelHeight * 3
- property color _textColor: "black"
- property real _rectOpacity: 0.8
- property var _guidedController: globals.guidedControllerFlyView
+ property real _margin: ScreenTools.defaultFontPixelWidth / 2
+ property real _widgetHeight: ScreenTools.defaultFontPixelHeight * 2
+ property var _guidedController: globals.guidedControllerFlyView
+ property var _activeVehicleColor: "green"
+ property var _activeVehicle: QGroundControl.multiVehicleManager.activeVehicle
+ property var selectedVehicles: QGroundControl.multiVehicleManager.selectedVehicles
QGCPalette { id: qgcPal }
- Rectangle {
- id: mvCommands
- anchors.left: parent.left
- anchors.right: parent.right
- height: mvCommandsColumn.height + (_margin *2)
- color: qgcPal.missionItemEditor
- opacity: _rectOpacity
- radius: _margin
-
- DeadMouseArea {
- anchors.fill: parent
+ function armAvailable() {
+ for (var i = 0; i < selectedVehicles.count; i++) {
+ var vehicle = selectedVehicles.get(i)
+ if (vehicle.armed === false) {
+ return true
+ }
}
+ return false
+ }
- Column {
- id: mvCommandsColumn
- anchors.margins: _margin
- anchors.top: parent.top
- anchors.left: parent.left
- anchors.right: parent.right
- spacing: _margin
-
- QGCLabel {
- anchors.left: parent.left
- anchors.right: parent.right
- text: qsTr("The following commands will be applied to all vehicles")
- color: _textColor
- wrapMode: Text.WordWrap
- font.pointSize: ScreenTools.smallFontPointSize
+
+ function disarmAvailable() {
+ for (var i = 0; i < selectedVehicles.count; i++) {
+ var vehicle = selectedVehicles.get(i)
+ if (vehicle.armed === true) {
+ return true
}
+ }
+ return false
+ }
- Row {
- spacing: _margin
+ function startAvailable() {
+ for (var i = 0; i < selectedVehicles.count; i++) {
+ var vehicle = selectedVehicles.get(i)
+ if (vehicle.armed === true && vehicle.flightMode !== vehicle.missionFlightMode){
+ return true
+ }
+ }
+ return false
+ }
- QGCButton {
- text: qsTr("Pause")
- onClicked: _guidedController.confirmAction(_guidedController.actionMVPause)
- }
+ function pauseAvailable() {
+ for (var i = 0; i < selectedVehicles.count; i++) {
+ var vehicle = selectedVehicles.get(i)
+ if (vehicle.armed === true && vehicle.pauseVehicleSupported) {
+ return true
+ }
+ }
+ return false
+ }
- QGCButton {
- text: qsTr("Start Mission")
- onClicked: _guidedController.confirmAction(_guidedController.actionMVStartMission)
- }
+ function selectVehicle(vehicleId) {
+ QGroundControl.multiVehicleManager.selectVehicle(vehicleId)
+ }
+
+ function deselectVehicle(vehicleId) {
+ QGroundControl.multiVehicleManager.deselectVehicle(vehicleId)
+ }
+
+ function toggleSelect(vehicleId) {
+ if (!vehicleSelected(vehicleId)) {
+ selectVehicle(vehicleId)
+ } else {
+ deselectVehicle(vehicleId)
+ }
+ }
+
+ function selectAll() {
+ var vehicles = QGroundControl.multiVehicleManager.vehicles
+ for (var i = 0; i < vehicles.count; i++) {
+ var vehicle = vehicles.get(i)
+ var vehicleId = vehicle.id
+ if (!vehicleSelected(vehicleId)) {
+ selectVehicle(vehicleId)
+ }
+ }
+ }
+
+ function deselectAll() {
+ QGroundControl.multiVehicleManager.deselectAllVehicles()
+ }
+
+ function vehicleSelected(vehicleId) {
+ for (var i = 0; i < selectedVehicles.count; i++ ) {
+ var currentId = selectedVehicles.get(i).id
+ if (vehicleId === currentId) {
+ return true
}
}
+ return false
}
QGCListView {
- id: missionItemEditorListView
+ id: vehicleList
anchors.left: parent.left
anchors.right: parent.right
anchors.topMargin: _margin
- anchors.top: mvCommands.bottom
+ anchors.top: parent.top
anchors.bottom: parent.bottom
spacing: ScreenTools.defaultFontPixelHeight / 2
orientation: ListView.Vertical
@@ -89,95 +126,91 @@ Item {
property real _cacheBuffer: height * 2
delegate: Rectangle {
- width: missionItemEditorListView.width
- height: innerColumn.y + innerColumn.height + _margin
- color: qgcPal.missionItemEditor
- opacity: _rectOpacity
- radius: _margin
+ width: vehicleList.width
+ height: innerColumn.y + innerColumn.height + _margin
+ color: QGroundControl.multiVehicleManager.activeVehicle == _vehicle ? _activeVehicleColor : qgcPal.button
+ radius: _margin
+ border.width: _vehicle && vehicleSelected(_vehicle.id) ? 1 : 0
+ border.color: qgcPal.text
property var _vehicle: object
- ColumnLayout {
- id: innerColumn
- anchors.margins: _margin
- anchors.top: parent.top
- anchors.left: parent.left
- anchors.right: parent.left
- spacing: _margin
+ Rectangle {
+ height: parent.height
+ width: innerColumn.width
+ anchors.horizontalCenter: parent.horizontalCenter
+ color: "transparent"
RowLayout {
- Layout.fillWidth: true
+ id: innerColumn
+ anchors.margins: _margin
+ spacing: _margin
- QGCLabel {
- Layout.alignment: Qt.AlignTop
- text: _vehicle ? _vehicle.id : ""
- color: _textColor
+ QGCCompassWidget {
+ id: compassWidget
+ size: _widgetHeight
+ usedByMultipleVehicleList: true
+ vehicle: _vehicle
}
- ColumnLayout {
- Layout.alignment: Qt.AlignCenter
- spacing: _margin
-
- FlightModeMenu {
- Layout.alignment: Qt.AlignHCenter
- font.pointSize: ScreenTools.largeFontPointSize
- color: _textColor
- currentVehicle: _vehicle
- }
-
- QGCLabel {
- Layout.alignment: Qt.AlignHCenter
- text: _vehicle && _vehicle.armed ? qsTr("Armed") : qsTr("Disarmed")
- color: _textColor
- }
+ QGCLabel {
+ text: " | "
+ font.pointSize: ScreenTools.largeFontPointSize
+ color: qgcPal.text
+ Layout.alignment: Qt.AlignHCenter
}
- QGCCompassWidget {
- size: _widgetHeight
- usedByMultipleVehicleList: true
- vehicle: _vehicle
+ QGCLabel {
+ text: _vehicle ? _vehicle.id : ""
+ font.pointSize: ScreenTools.largeFontPointSize
+ color: qgcPal.text
+ Layout.alignment: Qt.AlignHCenter
}
- QGCAttitudeWidget {
- size: _widgetHeight
- vehicle: _vehicle
+ QGCLabel {
+ text: " | "
+ font.pointSize: ScreenTools.largeFontPointSize
+ color: qgcPal.text
+ Layout.alignment: Qt.AlignHCenter
}
- } // RowLayout
- Row {
- spacing: ScreenTools.defaultFontPixelWidth
-
- QGCButton {
- text: qsTr("Arm")
- visible: _vehicle && !_vehicle.armed
- onClicked: _vehicle.armed = true
- }
+ ColumnLayout {
+ spacing: _margin
+ Layout.rightMargin: compassWidget.width / 4
+ Layout.alignment: Qt.AlignCenter
- QGCButton {
- text: qsTr("Start Mission")
- visible: _vehicle && _vehicle.armed && _vehicle.flightMode !== _vehicle.missionFlightMode
- onClicked: _vehicle.startMission()
- }
+ FlightModeMenu {
+ Layout.alignment: Qt.AlignHCenter
+ font.pointSize: ScreenTools.largeFontPointSize
+ color: qgcPal.text
+ currentVehicle: _vehicle
+ }
- QGCButton {
- text: qsTr("Pause")
- visible: _vehicle && _vehicle.armed && _vehicle.pauseVehicleSupported
- onClicked: _vehicle.pauseVehicle()
+ QGCLabel {
+ Layout.alignment: Qt.AlignHCenter
+ text: _vehicle && _vehicle.armed ? qsTr("Armed") : qsTr("Disarmed")
+ color: qgcPal.text
+ }
}
+ }
+ }
- QGCButton {
- text: qsTr("RTL")
- visible: _vehicle && _vehicle.armed && _vehicle.flightMode !== _vehicle.rtlFlightMode
- onClicked: _vehicle.flightMode = _vehicle.rtlFlightMode
- }
+ QGCMouseArea {
+ anchors.fill: parent
+ onClicked: singleClickTimer.start()
+ onDoubleClicked: {
+ singleClickTimer.stop()
+ QGroundControl.multiVehicleManager.activeVehicle = _vehicle
+ }
- QGCButton {
- text: qsTr("Take control")
- visible: _vehicle && _vehicle.armed && _vehicle.flightMode !== _vehicle.takeControlFlightMode
- onClicked: _vehicle.flightMode = _vehicle.takeControlFlightMode
- }
- } // Row
- } // ColumnLayout
- } // delegate - Rectangle
- } // QGCListView
-} // Item
+ Timer {
+ id: singleClickTimer
+ interval: 20
+ running: false
+ repeat: false
+ onTriggered: toggleSelect(_vehicle.id)
+ }
+ }
+ }
+ }
+}
diff --git a/src/FlightMap/Widgets/CompassHeadingIndicator.qml b/src/FlightMap/Widgets/CompassHeadingIndicator.qml
index 4be55228c2e..110f4172eca 100644
--- a/src/FlightMap/Widgets/CompassHeadingIndicator.qml
+++ b/src/FlightMap/Widgets/CompassHeadingIndicator.qml
@@ -23,6 +23,7 @@ Canvas {
property real compassSize
property real heading
+ property bool simplified: false
property var _qgcPal: QGroundControl.globalPalette
@@ -33,7 +34,7 @@ Canvas {
onPaint: {
var ctx = getContext("2d")
- ctx.strokeStyle = _qgcPal.text
+ ctx.strokeStyle = simplified ? "#EE3424" : _qgcPal.text
ctx.fillStyle = "#EE3424"
ctx.lineWidth = 1
ctx.beginPath()
diff --git a/src/FlightMap/Widgets/QGCCompassWidget.qml b/src/FlightMap/Widgets/QGCCompassWidget.qml
index 55ae9ad2b0a..18c81c2456e 100644
--- a/src/FlightMap/Widgets/QGCCompassWidget.qml
+++ b/src/FlightMap/Widgets/QGCCompassWidget.qml
@@ -21,12 +21,15 @@ Rectangle {
height: size
radius: width / 2
color: qgcPal.window
+ border.color: qgcPal.text
+ border.width: usedByMultipleVehicleList ? 1 : 0
+ opacity: vehicle && usedByMultipleVehicleList && !vehicle.armed ? 0.5 : 1
property real size: _defaultSize
property var vehicle: null
property bool usedByMultipleVehicleList: false
- property real _defaultSize: ScreenTools.defaultFontPixelHeight * (10)
+ property real _defaultSize: usedByMultipleVehicleList ? ScreenTools.defaultFontPixelHeight * 3 : ScreenTools.defaultFontPixelHeight * 10
property real _sizeRatio: ScreenTools.isTinyScreen ? (size / _defaultSize) * 0.5 : size / _defaultSize
property int _fontSize: ScreenTools.defaultFontPointSize * _sizeRatio < 8 ? 8 : ScreenTools.defaultFontPointSize * _sizeRatio
property real _heading: vehicle ? vehicle.heading.rawValue : 0
@@ -75,12 +78,14 @@ Rectangle {
}
CompassDial {
- anchors.fill: parent
+ anchors.fill: parent
+ visible: !usedByMultipleVehicleList
}
CompassHeadingIndicator {
compassSize: size
heading: _heading
+ simplified: usedByMultipleVehicleList
}
Image {
@@ -144,7 +149,7 @@ Rectangle {
QGCLabel {
anchors.horizontalCenter: parent.horizontalCenter
y: size * 0.74
- text: vehicle ? _heading.toFixed(0) + "°" : ""
+ text: vehicle && !usedByMultipleVehicleList ? _heading.toFixed(0) + "°" : ""
horizontalAlignment: Text.AlignHCenter
}
}
diff --git a/src/QmlControls/MvPanelPage.qml b/src/QmlControls/MvPanelPage.qml
new file mode 100644
index 00000000000..dd51a30b985
--- /dev/null
+++ b/src/QmlControls/MvPanelPage.qml
@@ -0,0 +1,33 @@
+import QtQuick
+import QtQuick.Controls
+
+import QGroundControl
+import QGroundControl.ScreenTools
+import QGroundControl.Palette
+
+Item {
+ property bool showBorder: true
+ property real contentMargin: _margins * 2
+ default property alias contentChildren: contentContainer.data
+
+ property real _margins: ScreenTools.defaultFontPixelHeight / 2
+
+ Rectangle {
+ color: "transparent"
+ x: _margins
+ y: _margins
+ height: parent.height - _margins * 2
+ width: parent.width - _margins * 2
+ border.color: QGroundControl.globalPalette.groupBorder
+ border.width: showBorder ? 1 : 0
+ radius: ScreenTools.defaultFontPixelHeight / 2
+
+
+ Item {
+ id: contentContainer
+ anchors.fill: parent
+ anchors.margins: _margins
+ }
+ }
+
+}
diff --git a/src/QmlControls/QGCPageIndicator.qml b/src/QmlControls/QGCPageIndicator.qml
new file mode 100644
index 00000000000..a4eef9bb535
--- /dev/null
+++ b/src/QmlControls/QGCPageIndicator.qml
@@ -0,0 +1,6 @@
+import QtQuick
+import QtQuick.Controls
+
+PageIndicator {
+
+}
diff --git a/src/QmlControls/QGCSwipeView.qml b/src/QmlControls/QGCSwipeView.qml
new file mode 100644
index 00000000000..f6fae4e9da7
--- /dev/null
+++ b/src/QmlControls/QGCSwipeView.qml
@@ -0,0 +1,7 @@
+import QtQuick
+import QtQuick.Controls
+
+
+SwipeView {
+
+}
diff --git a/src/QmlControls/QGroundControl/Controls/qmldir b/src/QmlControls/QGroundControl/Controls/qmldir
index 142d5eb1061..a37b67f7965 100644
--- a/src/QmlControls/QGroundControl/Controls/qmldir
+++ b/src/QmlControls/QGroundControl/Controls/qmldir
@@ -49,6 +49,9 @@ MissionItemEditor 1.0 MissionItemEditor.qml
MissionItemIndexLabel 1.0 MissionItemIndexLabel.qml
MissionItemMapVisual 1.0 MissionItemMapVisual.qml
MissionItemStatus 1.0 MissionItemStatus.qml
+ModeSwitchDisplay 1.0 ModeSwitchDisplay.qml
+MultiRotorMotorDisplay 1.0 MultiRotorMotorDisplay.qml
+MvPanelPage 1.0 MvPanelPage.qml
OfflineMapButton 1.0 OfflineMapButton.qml
OfflineMapEditor 1.0 OfflineMapEditor.qml
OfflineMapInfo 1.0 OfflineMapInfo.qml
@@ -84,11 +87,15 @@ QGCMenuSeparator 1.0 QGCMenuSeparator.qml
QGCMouseArea 1.0 QGCMouseArea.qml
QGCMovableItem 1.0 QGCMovableItem.qml
QGCOptionsComboBox 1.0 QGCOptionsComboBox.qml
+PipView 1.0 PipView.qml
+PipState 1.0 PipState.qml
+QGCPageIndicator 1.0 QGCPageIndicator.qml
QGCPopupDialog 1.0 QGCPopupDialog.qml
QGCRadioButton 1.0 QGCRadioButton.qml
QGCRoundButton 1.0 QGCRoundButton.qml
QGCSimpleMessageDialog 1.0 QGCSimpleMessageDialog.qml
QGCSlider 1.0 QGCSlider.qml
+QGCSwipeView 1.0 QGCSwipeView.qml
QGCSwitch 1.0 QGCSwitch.qml
QGCTabBar 1.0 QGCTabBar.qml
QGCTabButton 1.0 QGCTabButton.qml
diff --git a/src/QmlControls/QGroundControl/FlightDisplay/qmldir b/src/QmlControls/QGroundControl/FlightDisplay/qmldir
index 888f82ab0f1..79d0a3c817f 100644
--- a/src/QmlControls/QGroundControl/FlightDisplay/qmldir
+++ b/src/QmlControls/QGroundControl/FlightDisplay/qmldir
@@ -16,6 +16,7 @@ FlyViewToolBarIndicators 1.0 FlyViewToolBarIndicators.qml
FlyViewToolStrip 1.0 FlyViewToolStrip.qml
FlyViewToolStripActionList 1.0 FlyViewToolStripActionList.qml
FlyViewTopRightColumnLayout 1.0 FlyViewTopRightColumnLayout.qml
+FlyViewTopRightPanel 1.0 FlyViewTopRightPanel.qml
FlyViewVideo 1.0 FlyViewVideo.qml
FlyViewWidgetLayer 1.0 FlyViewWidgetLayer.qml
GripperMenu 1.0 GripperMenu.qml
diff --git a/src/Vehicle/MultiVehicleManager.cc b/src/Vehicle/MultiVehicleManager.cc
index 52cf5418f53..095767501b3 100644
--- a/src/Vehicle/MultiVehicleManager.cc
+++ b/src/Vehicle/MultiVehicleManager.cc
@@ -325,7 +325,7 @@ void MultiVehicleManager::_sendGCSHeartbeat()
MAV_MODE_MANUAL_ARMED,
0,
MAV_STATE_ACTIVE
- );
+ );
uint8_t buffer[MAVLINK_MAX_PACKET_LEN];
const uint16_t len = mavlink_msg_to_send_buffer(buffer, &message);
@@ -333,6 +333,41 @@ void MultiVehicleManager::_sendGCSHeartbeat()
}
}
+void MultiVehicleManager::selectVehicle(int vehicleId)
+{
+ if(!_vehicleSelected(vehicleId)) {
+ Vehicle *const vehicle = getVehicleById(vehicleId);
+ _selectedVehicles->append(vehicle);
+ }
+}
+
+void MultiVehicleManager::deselectVehicle(int vehicleId)
+{
+ for (int i = 0; i < _selectedVehicles->count(); i++) {
+ Vehicle *const vehicle = qobject_cast(_vehicles->get(i));
+ if (vehicle->id() == vehicleId) {
+ _selectedVehicles->removeAt(i);
+ return;
+ }
+ }
+}
+
+void MultiVehicleManager::deselectAllVehicles()
+{
+ _selectedVehicles->clear();
+}
+
+bool MultiVehicleManager::_vehicleSelected(int vehicleId)
+{
+ for (int i = 0; i < _selectedVehicles->count(); i++) {
+ Vehicle *const vehicle = qobject_cast(_vehicles->get(i));
+ if (vehicle->id() == vehicleId) {
+ return true;
+ }
+ }
+ return false;
+}
+
Vehicle *MultiVehicleManager::getVehicleById(int vehicleId) const
{
for (int i = 0; i < _vehicles->count(); i++) {
diff --git a/src/Vehicle/MultiVehicleManager.h b/src/Vehicle/MultiVehicleManager.h
index fc4cf5057d4..0cab1fa2969 100644
--- a/src/Vehicle/MultiVehicleManager.h
+++ b/src/Vehicle/MultiVehicleManager.h
@@ -33,6 +33,7 @@ class MultiVehicleManager : public QObject
Q_PROPERTY(bool parameterReadyVehicleAvailable READ _getParameterReadyVehicleAvailable NOTIFY parameterReadyVehicleAvailableChanged)
Q_PROPERTY(Vehicle *activeVehicle READ activeVehicle WRITE setActiveVehicle NOTIFY activeVehicleChanged)
Q_PROPERTY(QmlObjectListModel *vehicles READ vehicles CONSTANT)
+ Q_PROPERTY(QmlObjectListModel* selectedVehicles READ selectedVehicles CONSTANT)
Q_PROPERTY(bool gcsHeartBeatEnabled READ _getGcsHeartbeatEnabled WRITE _setGcsHeartbeatEnabled NOTIFY gcsHeartBeatEnabledChanged)
Q_PROPERTY(Vehicle *offlineEditingVehicle READ offlineEditingVehicle CONSTANT)
@@ -45,7 +46,11 @@ class MultiVehicleManager : public QObject
void init();
Q_INVOKABLE Vehicle *getVehicleById(int vehicleId) const;
+ Q_INVOKABLE void selectVehicle(int vehicleId);
+ Q_INVOKABLE void deselectVehicle(int vehicleId);
+ Q_INVOKABLE void deselectAllVehicles();
QmlObjectListModel *vehicles() const { return _vehicles; }
+ QmlObjectListModel *selectedVehicles(void) { return _selectedVehicles; }
Vehicle *offlineEditingVehicle() const { return _offlineEditingVehicle; }
Vehicle *activeVehicle() const { return _activeVehicle; }
void setActiveVehicle(Vehicle *vehicle);
@@ -69,6 +74,7 @@ private slots:
private:
bool _vehicleExists(int vehicleId);
+ bool _vehicleSelected(int vehicleId);
void _setActiveVehicle(Vehicle *vehicle);
bool _getGcsHeartbeatEnabled() const { return _gcsHeartbeatEnabled; }
void _setGcsHeartbeatEnabled(bool gcsHeartBeatEnabled);
@@ -80,6 +86,7 @@ private slots:
QTimer *_gcsHeartbeatTimer = nullptr; ///< Timer to emit heartbeats
Vehicle *_offlineEditingVehicle = nullptr; ///< Disconnected vechicle used for offline editing
QmlObjectListModel *_vehicles = nullptr;
+ QmlObjectListModel *_selectedVehicles = nullptr;
bool _activeVehicleAvailable = false; ///< true: An active vehicle is available
bool _gcsHeartbeatEnabled = false; ///< Enabled/disable heartbeat emission
bool _parameterReadyVehicleAvailable = false; ///< true: An active vehicle with ready parameters is available