地面站终端 App

1243 lines
56 KiB

/****************************************************************************
*
* (c) 2009-2020 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.
*
****************************************************************************/
import QtQuick 2.3
import QtQuick.Controls 1.2
import QtQuick.Dialogs 1.2
import QtLocation 5.3
import QtPositioning 5.3
import QtQuick.Layouts 1.2
import QtQuick.Window 2.2
import QGroundControl 1.0
import QGroundControl.FlightMap 1.0
import QGroundControl.ScreenTools 1.0
import QGroundControl.Controls 1.0
import QGroundControl.FactSystem 1.0
import QGroundControl.FactControls 1.0
import QGroundControl.Palette 1.0
import QGroundControl.Controllers 1.0
import QGroundControl.ShapeFileHelper 1.0
import QGroundControl.Airspace 1.0
import QGroundControl.Airmap 1.0
Item {
id: _root
property bool planControlColapsed: false
readonly property int _decimalPlaces: 8
readonly property real _margin: ScreenTools.defaultFontPixelHeight * 0.5
readonly property real _toolsMargin: ScreenTools.defaultFontPixelWidth * 0.75
readonly property real _radius: ScreenTools.defaultFontPixelWidth * 0.5
readonly property real _rightPanelWidth: Math.min(parent.width / 3, ScreenTools.defaultFontPixelWidth * 30)
readonly property var _defaultVehicleCoordinate: QtPositioning.coordinate(37.803784, -122.462276)
readonly property bool _waypointsOnlyMode: QGroundControl.corePlugin.options.missionWaypointsOnly
property bool _airspaceEnabled: QGroundControl.airmapSupported ? (QGroundControl.settingsManager.airMapSettings.enableAirMap.rawValue && QGroundControl.airspaceManager.connected): false
property var _missionController: _planMasterController.missionController
property var _geoFenceController: _planMasterController.geoFenceController
property var _rallyPointController: _planMasterController.rallyPointController
property var _visualItems: _missionController.visualItems
property bool _lightWidgetBorders: editorMap.isSatelliteMap
property bool _addWaypointOnClick: false
property bool _addROIOnClick: false
property bool _singleComplexItem: _missionController.complexMissionItemNames.length === 1
property int _editingLayer: layerTabBar.currentIndex ? _layers[layerTabBar.currentIndex] : _layerMission
property int _toolStripBottom: toolStrip.height + toolStrip.y
property var _appSettings: QGroundControl.settingsManager.appSettings
property var _planViewSettings: QGroundControl.settingsManager.planViewSettings
property bool _promptForPlanUsageShowing: false
readonly property var _layers: [_layerMission, _layerGeoFence, _layerRallyPoints]
readonly property int _layerMission: 1
readonly property int _layerGeoFence: 2
readonly property int _layerRallyPoints: 3
readonly property string _armedVehicleUploadPrompt: qsTr("Vehicle is currently armed. Do you want to upload the mission to the vehicle?")
function mapCenter() {
var coordinate = editorMap.center
coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
return coordinate
}
function updateAirspace(reset) {
if(_airspaceEnabled) {
var coordinateNW = editorMap.toCoordinate(Qt.point(0,0), false /* clipToViewPort */)
var coordinateSE = editorMap.toCoordinate(Qt.point(width,height), false /* clipToViewPort */)
if(coordinateNW.isValid && coordinateSE.isValid) {
QGroundControl.airspaceManager.setROI(coordinateNW, coordinateSE, true /*planView*/, reset)
}
}
}
property bool _firstMissionLoadComplete: false
property bool _firstFenceLoadComplete: false
property bool _firstRallyLoadComplete: false
property bool _firstLoadComplete: false
MapFitFunctions {
id: mapFitFunctions // The name for this id cannot be changed without breaking references outside of this code. Beware!
map: editorMap
usePlannedHomePosition: true
planMasterController: _planMasterController
}
on_AirspaceEnabledChanged: {
if(QGroundControl.airmapSupported) {
if(_airspaceEnabled) {
planControlColapsed = QGroundControl.airspaceManager.airspaceVisible
updateAirspace(true)
} else {
planControlColapsed = false
}
} else {
planControlColapsed = false
}
}
onVisibleChanged: {
if(visible) {
editorMap.zoomLevel = QGroundControl.flightMapZoom
editorMap.center = QGroundControl.flightMapPosition
if (!_planMasterController.containsItems) {
toolStrip.simulateClick(toolStrip.fileButtonIndex)
}
}
}
Connections {
target: _appSettings ? _appSettings.defaultMissionItemAltitude : null
onRawValueChanged: {
if (_visualItems.count > 1) {
mainWindow.showComponentDialog(applyNewAltitude, qsTr("Apply new alititude"), mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
}
}
}
Component {
id: applyNewAltitude
QGCViewMessage {
message: qsTr("You have changed the default altitude for mission items. Would you like to apply that altitude to all the items in the current mission?")
function accept() {
hideDialog()
_missionController.applyDefaultMissionAltitude()
}
}
}
Component {
id: promptForPlanUsageOnVehicleChangePopupComponent
QGCPopupDialog {
title: _planMasterController.managerVehicle.isOfflineEditingVehicle ? qsTr("Plan View - Vehicle Disconnected") : qsTr("Plan View - Vehicle Changed")
buttons: StandardButton.NoButton
ColumnLayout {
QGCLabel {
Layout.maximumWidth: parent.width
wrapMode: QGCLabel.WordWrap
text: _planMasterController.managerVehicle.isOfflineEditingVehicle ?
qsTr("The vehicle associated with the plan in the Plan View is no longer available. What would you like to do with that plan?") :
qsTr("The plan being worked on in the Plan View is not from the current vehicle. What would you like to do with that plan?")
}
QGCButton {
Layout.fillWidth: true
text: _planMasterController.dirty ?
(_planMasterController.managerVehicle.isOfflineEditingVehicle ?
qsTr("Discard Unsaved Changes") :
qsTr("Discard Unsaved Changes, Load New Plan From Vehicle")) :
qsTr("Load New Plan From Vehicle")
onClicked: {
_planMasterController.showPlanFromManagerVehicle()
_promptForPlanUsageShowing = false
hideDialog();
}
}
QGCButton {
Layout.fillWidth: true
text: _planMasterController.managerVehicle.isOfflineEditingVehicle ?
qsTr("Keep Current Plan") :
qsTr("Keep Current Plan, Don't Update From Vehicle")
onClicked: {
if (!_planMasterController.managerVehicle.isOfflineEditingVehicle) {
_planMasterController.dirty = true
}
_promptForPlanUsageShowing = false
hideDialog()
}
}
}
}
}
Component {
id: firmwareOrVehicleMismatchUploadDialogComponent
QGCViewMessage {
message: qsTr("This Plan was created for a different firmware or vehicle type than the firmware/vehicle type of vehicle you are uploading to. " +
"This can lead to errors or incorrect behavior. " +
"It is recommended to recreate the Plan for the correct firmware/vehicle type.\n\n" +
"Click 'Ok' to upload the Plan anyway.")
function accept() {
_planMasterController.sendToVehicle()
hideDialog()
}
}
}
Connections {
target: QGroundControl.airspaceManager
onAirspaceVisibleChanged: {
planControlColapsed = QGroundControl.airspaceManager.airspaceVisible
}
}
Component {
id: noItemForKML
QGCViewMessage {
message: qsTr("You need at least one item to create a KML.")
}
}
PlanMasterController {
id: _planMasterController
flyView: false
Component.onCompleted: {
_planMasterController.start()
_missionController.setCurrentPlanViewSeqNum(0, true)
globals.planMasterControllerPlanView = _planMasterController
}
onPromptForPlanUsageOnVehicleChange: {
if (!_promptForPlanUsageShowing) {
_promptForPlanUsageShowing = true
mainWindow.showPopupDialogFromComponent(promptForPlanUsageOnVehicleChangePopupComponent)
}
}
function waitingOnIncompleteDataMessage(save) {
var saveOrUpload = save ? qsTr("Save") : qsTr("Upload")
mainWindow.showMessageDialog(qsTr("Unable to %1").arg(saveOrUpload), qsTr("Plan has incomplete items. Complete all items and %1 again.").arg(saveOrUpload))
}
function waitingOnTerrainDataMessage(save) {
var saveOrUpload = save ? qsTr("Save") : qsTr("Upload")
mainWindow.showMessageDialog(qsTr("Unable to %1").arg(saveOrUpload), qsTr("Plan is waiting on terrain data from server for correct altitude values."))
}
function checkReadyForSaveUpload(save) {
if (readyForSaveState() == VisualMissionItem.NotReadyForSaveData) {
waitingOnIncompleteDataMessage(save)
return false
} else if (readyForSaveState() == VisualMissionItem.NotReadyForSaveTerrain) {
waitingOnTerrainDataMessage(save)
return false
}
return true
}
function upload() {
if (!checkReadyForSaveUpload(false /* save */)) {
return
}
switch (_missionController.sendToVehiclePreCheck()) {
case MissionController.SendToVehiclePreCheckStateOk:
sendToVehicle()
break
case MissionController.SendToVehiclePreCheckStateActiveMission:
mainWindow.showMessageDialog(qsTr("Send To Vehicle"), qsTr("Current mission must be paused prior to uploading a new Plan"))
break
case MissionController.SendToVehiclePreCheckStateFirwmareVehicleMismatch:
mainWindow.showComponentDialog(firmwareOrVehicleMismatchUploadDialogComponent, qsTr("Plan Upload"), mainWindow.showDialogDefaultWidth, StandardButton.Ok | StandardButton.Cancel)
break
}
}
function loadFromSelectedFile() {
fileDialog.title = qsTr("Select Plan File")
fileDialog.planFiles = true
fileDialog.selectExisting = true
fileDialog.nameFilters = _planMasterController.loadNameFilters
fileDialog.openForLoad()
}
function saveToSelectedFile() {
if (!checkReadyForSaveUpload(true /* save */)) {
return
}
fileDialog.title = qsTr("Save Plan")
fileDialog.planFiles = true
fileDialog.selectExisting = false
fileDialog.nameFilters = _planMasterController.saveNameFilters
fileDialog.openForSave()
}
function fitViewportToItems() {
mapFitFunctions.fitMapViewportToMissionItems()
}
function saveKmlToSelectedFile() {
if (!checkReadyForSaveUpload(true /* save */)) {
return
}
fileDialog.title = qsTr("Save KML")
fileDialog.planFiles = false
fileDialog.selectExisting = false
fileDialog.nameFilters = ShapeFileHelper.fileDialogKMLFilters
fileDialog.openForSave()
}
}
Connections {
target: _missionController
onNewItemsFromVehicle: {
if (_visualItems && _visualItems.count !== 1) {
mapFitFunctions.fitMapViewportToMissionItems()
}
_missionController.setCurrentPlanViewSeqNum(0, true)
}
}
function insertSimpleItemAfterCurrent(coordinate) {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertSimpleMissionItem(coordinate, nextIndex, true /* makeCurrentItem */)
}
function insertROIAfterCurrent(coordinate) {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertROIMissionItem(coordinate, nextIndex, true /* makeCurrentItem */)
}
function insertCancelROIAfterCurrent() {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertCancelROIMissionItem(nextIndex, true /* makeCurrentItem */)
}
function insertComplexItemAfterCurrent(complexItemName) {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertComplexMissionItem(complexItemName, mapCenter(), nextIndex, true /* makeCurrentItem */)
}
function insertTakeItemAfterCurrent() {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertTakeoffItem(mapCenter(), nextIndex, true /* makeCurrentItem */)
}
function insertLandItemAfterCurrent() {
var nextIndex = _missionController.currentPlanViewVIIndex + 1
_missionController.insertLandItem(mapCenter(), nextIndex, true /* makeCurrentItem */)
}
function selectNextNotReady() {
var foundCurrent = false
for (var i=0; i<_missionController.visualItems.count; i++) {
var vmi = _missionController.visualItems.get(i)
if (vmi.readyForSaveState === VisualMissionItem.NotReadyForSaveData) {
_missionController.setCurrentPlanViewSeqNum(vmi.sequenceNumber, true)
break
}
}
}
property int _moveDialogMissionItemIndex
QGCFileDialog {
id: fileDialog
folder: _appSettings ? _appSettings.missionSavePath : ""
property bool planFiles: true ///< true: working with plan files, false: working with kml file
onAcceptedForSave: {
if (planFiles) {
_planMasterController.saveToFile(file)
} else {
_planMasterController.saveToKml(file)
}
close()
}
onAcceptedForLoad: {
_planMasterController.loadFromFile(file)
_planMasterController.fitViewportToItems()
_missionController.setCurrentPlanViewSeqNum(0, true)
close()
}
}
Component {
id: moveDialog
QGCViewDialog {
function accept() {
var toIndex = toCombo.currentIndex
if (toIndex === 0) {
toIndex = 1
}
_missionController.moveMissionItem(_moveDialogMissionItemIndex, toIndex)
hideDialog()
}
Column {
anchors.left: parent.left
anchors.right: parent.right
spacing: ScreenTools.defaultFontPixelHeight
QGCLabel {
anchors.left: parent.left
anchors.right: parent.right
wrapMode: Text.WordWrap
text: qsTr("Move the selected mission item to the be after following mission item:")
}
QGCComboBox {
id: toCombo
model: _visualItems.count
currentIndex: _moveDialogMissionItemIndex
}
}
}
}
Item {
id: panel
anchors.fill: parent
FlightMap {
id: editorMap
anchors.fill: parent
mapName: "MissionEditor"
allowGCSLocationCenter: true
allowVehicleLocationCenter: true
planView: true
zoomLevel: QGroundControl.flightMapZoom
center: QGroundControl.flightMapPosition
// This is the center rectangle of the map which is not obscured by tools
property rect centerViewport: Qt.rect(_leftToolWidth + _margin, _margin, editorMap.width - _leftToolWidth - _rightToolWidth - (_margin * 2), (terrainStatus.visible ? terrainStatus.y : height - _margin) - _margin)
property real _leftToolWidth: toolStrip.x + toolStrip.width
property real _rightToolWidth: rightPanel.width + rightPanel.anchors.rightMargin
property real _nonInteractiveOpacity: 0.5
// Initial map position duplicates Fly view position
Component.onCompleted: editorMap.center = QGroundControl.flightMapPosition
QGCMapPalette { id: mapPal; lightColors: editorMap.isSatelliteMap }
onZoomLevelChanged: {
QGroundControl.flightMapZoom = zoomLevel
updateAirspace(false)
}
onCenterChanged: {
QGroundControl.flightMapPosition = center
updateAirspace(false)
}
MouseArea {
anchors.fill: parent
onClicked: {
// Take focus to close any previous editing
editorMap.focus = true
var coordinate = editorMap.toCoordinate(Qt.point(mouse.x, mouse.y), false /* clipToViewPort */)
coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
switch (_editingLayer) {
case _layerMission:
if (_addWaypointOnClick) {
insertSimpleItemAfterCurrent(coordinate)
} else if (_addROIOnClick) {
insertROIAfterCurrent(coordinate)
_addROIOnClick = false
}
break
case _layerRallyPoints:
if (_rallyPointController.supported && _addWaypointOnClick) {
_rallyPointController.addPoint(coordinate)
}
break
}
}
}
// Add the mission item visuals to the map
Repeater {
model: _missionController.visualItems
delegate: MissionItemMapVisual {
map: editorMap
onClicked: _missionController.setCurrentPlanViewSeqNum(sequenceNumber, false)
opacity: _editingLayer == _layerMission ? 1 : editorMap._nonInteractiveOpacity
interactive: _editingLayer == _layerMission
}
}
// Add lines between waypoints
MissionLineView {
showSpecialVisual: _missionController.isROIBeginCurrentItem
model: _missionController.simpleFlightPathSegments
opacity: _editingLayer == _layerMission ? 1 : editorMap._nonInteractiveOpacity
}
// Direction arrows in waypoint lines
MapItemView {
model: _editingLayer == _layerMission ? _missionController.directionArrows : undefined
delegate: MapLineArrow {
fromCoord: object ? object.coordinate1 : undefined
toCoord: object ? object.coordinate2 : undefined
arrowPosition: 3
z: QGroundControl.zOrderWaypointLines + 1
}
}
// Incomplete segment lines
MapItemView {
model: _missionController.incompleteComplexItemLines
delegate: MapPolyline {
path: [ object.coordinate1, object.coordinate2 ]
line.width: 1
line.color: "red"
z: QGroundControl.zOrderWaypointLines
opacity: _editingLayer == _layerMission ? 1 : editorMap._nonInteractiveOpacity
}
}
// UI for splitting the current segment
MapQuickItem {
id: splitSegmentItem
anchorPoint.x: sourceItem.width / 2
anchorPoint.y: sourceItem.height / 2
z: QGroundControl.zOrderWaypointLines + 1
visible: _editingLayer == _layerMission
sourceItem: SplitIndicator {
onClicked: _missionController.insertSimpleMissionItem(splitSegmentItem.coordinate,
_missionController.currentPlanViewVIIndex,
true /* makeCurrentItem */)
}
function _updateSplitCoord() {
if (_missionController.splitSegment) {
var distance = _missionController.splitSegment.coordinate1.distanceTo(_missionController.splitSegment.coordinate2)
var azimuth = _missionController.splitSegment.coordinate1.azimuthTo(_missionController.splitSegment.coordinate2)
splitSegmentItem.coordinate = _missionController.splitSegment.coordinate1.atDistanceAndAzimuth(distance / 2, azimuth)
} else {
coordinate = QtPositioning.coordinate()
}
}
Connections {
target: _missionController
onSplitSegmentChanged: splitSegmentItem._updateSplitCoord()
}
Connections {
target: _missionController.splitSegment
onCoordinate1Changed: splitSegmentItem._updateSplitCoord()
onCoordinate2Changed: splitSegmentItem._updateSplitCoord()
}
}
// Add the vehicles to the map
MapItemView {
model: QGroundControl.multiVehicleManager.vehicles
delegate: VehicleMapItem {
vehicle: object
coordinate: object.coordinate
map: editorMap
size: ScreenTools.defaultFontPixelHeight * 3
z: QGroundControl.zOrderMapItems - 1
}
}
GeoFenceMapVisuals {
map: editorMap
myGeoFenceController: _geoFenceController
interactive: _editingLayer == _layerGeoFence
homePosition: _missionController.plannedHomePosition
planView: true
opacity: _editingLayer != _layerGeoFence ? editorMap._nonInteractiveOpacity : 1
}
RallyPointMapVisuals {
map: editorMap
myRallyPointController: _rallyPointController
interactive: _editingLayer == _layerRallyPoints
planView: true
opacity: _editingLayer != _layerRallyPoints ? editorMap._nonInteractiveOpacity : 1
}
// Airspace overlap support
MapItemView {
model: _airspaceEnabled && QGroundControl.airspaceManager.airspaceVisible ? QGroundControl.airspaceManager.airspaces.circles : []
delegate: MapCircle {
center: object.center
radius: object.radius
color: object.color
border.color: object.lineColor
border.width: object.lineWidth
}
}
MapItemView {
model: _airspaceEnabled && QGroundControl.airspaceManager.airspaceVisible ? QGroundControl.airspaceManager.airspaces.polygons : []
delegate: MapPolygon {
path: object.polygon
color: object.color
border.color: object.lineColor
border.width: object.lineWidth
}
}
}
//-----------------------------------------------------------
// Left tool strip
ToolStrip {
id: toolStrip
anchors.margins: _toolsMargin
anchors.left: parent.left
anchors.top: parent.top
z: QGroundControl.zOrderWidgets
maxHeight: parent.height - toolStrip.y
title: qsTr("Plan")
readonly property int flyButtonIndex: 0
readonly property int fileButtonIndex: 1
readonly property int takeoffButtonIndex: 2
readonly property int waypointButtonIndex: 3
readonly property int roiButtonIndex: 4
readonly property int patternButtonIndex: 5
readonly property int landButtonIndex: 6
readonly property int centerButtonIndex: 7
property bool _isRallyLayer: _editingLayer == _layerRallyPoints
property bool _isMissionLayer: _editingLayer == _layerMission
ToolStripActionList {
id: toolStripActionList
model: [
ToolStripAction {
text: qsTr("Fly")
iconSource: "/qmlimages/PaperPlane.svg"
onTriggered: mainWindow.showFlyView()
},
ToolStripAction {
text: qsTr("File")
enabled: !_planMasterController.syncInProgress
visible: true
showAlternateIcon: _planMasterController.dirty
iconSource: "/qmlimages/MapSync.svg"
alternateIconSource: "/qmlimages/MapSyncChanged.svg"
dropPanelComponent: syncDropPanel
},
ToolStripAction {
text: qsTr("Takeoff")
iconSource: "/res/takeoff.svg"
enabled: _missionController.isInsertTakeoffValid
visible: toolStrip._isMissionLayer && !_planMasterController.controllerVehicle.rover
onTriggered: {
toolStrip.allAddClickBoolsOff()
insertTakeItemAfterCurrent()
}
},
ToolStripAction {
text: _editingLayer == _layerRallyPoints ? qsTr("Rally Point") : qsTr("Waypoint")
iconSource: "/qmlimages/MapAddMission.svg"
enabled: toolStrip._isRallyLayer ? true : _missionController.flyThroughCommandsAllowed
visible: toolStrip._isRallyLayer || toolStrip._isMissionLayer
checkable: true
onCheckedChanged: _addWaypointOnClick = checked
property bool myAddWaypointOnClick: _addWaypointOnClick
onMyAddWaypointOnClickChanged: checked = _addWaypointOnClick
},
ToolStripAction {
text: _missionController.isROIActive ? qsTr("Cancel ROI") : qsTr("ROI")
iconSource: "/qmlimages/MapAddMission.svg"
enabled: !_missionController.onlyInsertTakeoffValid
visible: toolStrip._isMissionLayer && _planMasterController.controllerVehicle.roiModeSupported
checkable: !_missionController.isROIActive
onCheckedChanged: _addROIOnClick = checked
onTriggered: {
if (_missionController.isROIActive) {
toolStrip.allAddClickBoolsOff()
insertCancelROIAfterCurrent()
}
}
property bool myAddROIOnClick: _addROIOnClick
onMyAddROIOnClickChanged: checked = _addROIOnClick
},
ToolStripAction {
text: _singleComplexItem ? _missionController.complexMissionItemNames[0] : qsTr("Pattern")
iconSource: "/qmlimages/MapDrawShape.svg"
enabled: _missionController.flyThroughCommandsAllowed
visible: toolStrip._isMissionLayer
dropPanelComponent: _singleComplexItem ? undefined : patternDropPanel
onTriggered: {
toolStrip.allAddClickBoolsOff()
if (_singleComplexItem) {
insertComplexItemAfterCurrent(_missionController.complexMissionItemNames[0])
}
}
},
ToolStripAction {
text: _planMasterController.controllerVehicle.multiRotor ? qsTr("Return") : qsTr("Land")
iconSource: "/res/rtl.svg"
enabled: _missionController.isInsertLandValid
visible: toolStrip._isMissionLayer
onTriggered: {
toolStrip.allAddClickBoolsOff()
insertLandItemAfterCurrent()
}
},
ToolStripAction {
text: qsTr("Center")
iconSource: "/qmlimages/MapCenter.svg"
enabled: true
visible: true
dropPanelComponent: centerMapDropPanel
}
]
}
model: toolStripActionList.model
function allAddClickBoolsOff() {
_addROIOnClick = false
_addWaypointOnClick = false
}
onDropped: allAddClickBoolsOff()
}
//-----------------------------------------------------------
// Right pane for mission editing controls
Rectangle {
id: rightPanel
height: parent.height
width: _rightPanelWidth
color: qgcPal.window
opacity: layerTabBar.visible ? 0.2 : 0
anchors.bottom: parent.bottom
anchors.right: parent.right
anchors.rightMargin: _toolsMargin
}
//-------------------------------------------------------
// Right Panel Controls
Item {
anchors.fill: rightPanel
anchors.topMargin: _toolsMargin
DeadMouseArea {
anchors.fill: parent
}
Column {
id: rightControls
spacing: ScreenTools.defaultFontPixelHeight * 0.5
anchors.left: parent.left
anchors.right: parent.right
anchors.top: parent.top
//-------------------------------------------------------
// Airmap Airspace Control
AirspaceControl {
id: airspaceControl
width: parent.width
visible: _airspaceEnabled
planView: true
showColapse: true
}
//-------------------------------------------------------
// Mission Controls (Colapsed)
Rectangle {
width: parent.width
height: planControlColapsed ? colapsedRow.height + ScreenTools.defaultFontPixelHeight : 0
color: qgcPal.missionItemEditor
radius: _radius
visible: planControlColapsed && _airspaceEnabled
Row {
id: colapsedRow
spacing: ScreenTools.defaultFontPixelWidth
anchors.left: parent.left
anchors.leftMargin: ScreenTools.defaultFontPixelWidth
anchors.verticalCenter: parent.verticalCenter
QGCColoredImage {
width: height
height: ScreenTools.defaultFontPixelWidth * 2.5
sourceSize.height: height
source: "qrc:/res/waypoint.svg"
color: qgcPal.text
anchors.verticalCenter: parent.verticalCenter
}
QGCLabel {
text: qsTr("Plan")
color: qgcPal.text
anchors.verticalCenter: parent.verticalCenter
}
}
QGCColoredImage {
width: height
height: ScreenTools.defaultFontPixelWidth * 2.5
sourceSize.height: height
source: QGroundControl.airmapSupported ? "qrc:/airmap/expand.svg" : ""
color: "white"
visible: QGroundControl.airmapSupported
anchors.right: parent.right
anchors.rightMargin: ScreenTools.defaultFontPixelWidth
anchors.verticalCenter: parent.verticalCenter
}
MouseArea {
anchors.fill: parent
enabled: QGroundControl.airmapSupported
onClicked: {
QGroundControl.airspaceManager.airspaceVisible = false
}
}
}
//-------------------------------------------------------
// Mission Controls (Expanded)
QGCTabBar {
id: layerTabBar
width: parent.width
visible: (!planControlColapsed || !_airspaceEnabled) && QGroundControl.corePlugin.options.enablePlanViewSelector
Component.onCompleted: currentIndex = 0
QGCTabButton {
text: qsTr("Mission")
}
QGCTabButton {
text: qsTr("Fence")
enabled: _geoFenceController.supported
}
QGCTabButton {
text: qsTr("Rally")
enabled: _rallyPointController.supported
}
}
}
//-------------------------------------------------------
// Mission Item Editor
Item {
id: missionItemEditor
anchors.left: parent.left
anchors.right: parent.right
anchors.top: rightControls.bottom
anchors.topMargin: ScreenTools.defaultFontPixelHeight * 0.25
anchors.bottom: parent.bottom
anchors.bottomMargin: ScreenTools.defaultFontPixelHeight * 0.25
visible: _editingLayer == _layerMission && !planControlColapsed
QGCListView {
id: missionItemEditorListView
anchors.fill: parent
spacing: ScreenTools.defaultFontPixelHeight / 4
orientation: ListView.Vertical
model: _missionController.visualItems
cacheBuffer: Math.max(height * 2, 0)
clip: true
currentIndex: _missionController.currentPlanViewSeqNum
highlightMoveDuration: 250
visible: _editingLayer == _layerMission && !planControlColapsed
//-- List Elements
delegate: MissionItemEditor {
map: editorMap
masterController: _planMasterController
missionItem: object
width: parent.width
readOnly: false
onClicked: _missionController.setCurrentPlanViewSeqNum(object.sequenceNumber, false)
onRemove: {
var removeVIIndex = index
_missionController.removeVisualItem(removeVIIndex)
if (removeVIIndex >= _missionController.visualItems.count) {
removeVIIndex--
}
}
onSelectNextNotReadyItem: selectNextNotReady()
}
}
}
// GeoFence Editor
GeoFenceEditor {
anchors.top: rightControls.bottom
anchors.topMargin: ScreenTools.defaultFontPixelHeight * 0.25
anchors.bottom: parent.bottom
anchors.left: parent.left
anchors.right: parent.right
myGeoFenceController: _geoFenceController
flightMap: editorMap
visible: _editingLayer == _layerGeoFence
}
// Rally Point Editor
RallyPointEditorHeader {
id: rallyPointHeader
anchors.top: rightControls.bottom
anchors.topMargin: ScreenTools.defaultFontPixelHeight * 0.25
anchors.left: parent.left
anchors.right: parent.right
visible: _editingLayer == _layerRallyPoints
controller: _rallyPointController
}
RallyPointItemEditor {
id: rallyPointEditor
anchors.top: rallyPointHeader.bottom
anchors.topMargin: ScreenTools.defaultFontPixelHeight * 0.25
anchors.left: parent.left
anchors.right: parent.right
visible: _editingLayer == _layerRallyPoints && _rallyPointController.points.count
rallyPoint: _rallyPointController.currentRallyPoint
controller: _rallyPointController
}
}
TerrainStatus {
id: terrainStatus
anchors.margins: _toolsMargin
anchors.leftMargin: 0
anchors.left: mapScale.left
anchors.right: rightPanel.left
anchors.bottom: parent.bottom
height: ScreenTools.defaultFontPixelHeight * 7
missionController: _missionController
visible: _internalVisible && _editingLayer === _layerMission && QGroundControl.corePlugin.options.showMissionStatus
onSetCurrentSeqNum: _missionController.setCurrentPlanViewSeqNum(seqNum, true)
property bool _internalVisible: _planViewSettings.showMissionItemStatus.rawValue
function toggleVisible() {
_internalVisible = !_internalVisible
_planViewSettings.showMissionItemStatus.rawValue = _internalVisible
}
}
MapScale {
id: mapScale
anchors.margins: _toolsMargin
anchors.bottom: terrainStatus.visible ? terrainStatus.top : parent.bottom
anchors.left: toolStrip.y + toolStrip.height + _toolsMargin > mapScale.y ? toolStrip.right: parent.left
mapControl: editorMap
buttonsOnLeft: true
terrainButtonVisible: _editingLayer === _layerMission
terrainButtonChecked: terrainStatus.visible
onTerrainButtonClicked: terrainStatus.toggleVisible()
}
}
Component {
id: syncLoadFromVehicleOverwrite
QGCViewMessage {
id: syncLoadFromVehicleCheck
message: qsTr("You have unsaved/unsent changes. Loading from the Vehicle will lose these changes. Are you sure you want to load from the Vehicle?")
function accept() {
hideDialog()
_planMasterController.loadFromVehicle()
}
}
}
Component {
id: syncLoadFromFileOverwrite
QGCViewMessage {
id: syncLoadFromVehicleCheck
message: qsTr("You have unsaved/unsent changes. Loading from a file will lose these changes. Are you sure you want to load from a file?")
function accept() {
hideDialog()
_planMasterController.loadFromSelectedFile()
}
}
}
property var createPlanRemoveAllPromptDialogMapCenter
property var createPlanRemoveAllPromptDialogPlanCreator
Component {
id: createPlanRemoveAllPromptDialog
QGCViewMessage {
message: qsTr("Are you sure you want to remove current plan and create a new plan? ")
function accept() {
createPlanRemoveAllPromptDialogPlanCreator.createPlan(createPlanRemoveAllPromptDialogMapCenter)
hideDialog()
}
}
}
Component {
id: clearVehicleMissionDialog
QGCViewMessage {
message: qsTr("Are you sure you want to remove all mission items and clear the mission from the vehicle?")
function accept() {
_planMasterController.removeAllFromVehicle()
_missionController.setCurrentPlanViewSeqNum(0, true)
hideDialog()
}
}
}
//- ToolStrip DropPanel Components
Component {
id: centerMapDropPanel
CenterMapDropPanel {
map: editorMap
fitFunctions: mapFitFunctions
}
}
Component {
id: patternDropPanel
ColumnLayout {
spacing: ScreenTools.defaultFontPixelWidth * 0.5
QGCLabel { text: qsTr("Create complex pattern:") }
Repeater {
model: _missionController.complexMissionItemNames
QGCButton {
text: modelData
Layout.fillWidth: true
onClicked: {
insertComplexItemAfterCurrent(modelData)
dropPanel.hide()
}
}
}
} // Column
}
Component {
id: syncDropPanel
ColumnLayout {
id: columnHolder
spacing: _margin
property string _overwriteText: (_editingLayer == _layerMission) ? qsTr("Mission overwrite") : ((_editingLayer == _layerGeoFence) ? qsTr("GeoFence overwrite") : qsTr("Rally Points overwrite"))
QGCLabel {
id: unsavedChangedLabel
Layout.fillWidth: true
wrapMode: Text.WordWrap
text: globals.activeVehicle ?
qsTr("You have unsaved changes. You should upload to your vehicle, or save to a file.") :
qsTr("You have unsaved changes.")
visible: _planMasterController.dirty
}
SectionHeader {
id: createSection
Layout.fillWidth: true
text: qsTr("Create Plan")
showSpacer: false
}
GridLayout {
columns: 2
columnSpacing: _margin
rowSpacing: _margin
Layout.fillWidth: true
visible: createSection.visible
Repeater {
model: _planMasterController.planCreators
Rectangle {
id: button
width: ScreenTools.defaultFontPixelHeight * 7
height: planCreatorNameLabel.y + planCreatorNameLabel.height
color: button.pressed || button.highlighted ? qgcPal.buttonHighlight : qgcPal.button
property bool highlighted: mouseArea.containsMouse
property bool pressed: mouseArea.pressed
Image {
id: planCreatorImage
anchors.left: parent.left
anchors.right: parent.right
source: object.imageResource
sourceSize.width: width
fillMode: Image.PreserveAspectFit
mipmap: true
}
QGCLabel {
id: planCreatorNameLabel
anchors.top: planCreatorImage.bottom
anchors.left: parent.left
anchors.right: parent.right
horizontalAlignment: Text.AlignHCenter
text: object.name
color: button.pressed || button.highlighted ? qgcPal.buttonHighlightText : qgcPal.buttonText
}
QGCMouseArea {
id: mouseArea
anchors.fill: parent
hoverEnabled: true
preventStealing: true
onClicked: {
if (_planMasterController.containsItems) {
createPlanRemoveAllPromptDialogMapCenter = _mapCenter()
createPlanRemoveAllPromptDialogPlanCreator = object
mainWindow.showComponentDialog(createPlanRemoveAllPromptDialog, qsTr("Create Plan"), mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
} else {
object.createPlan(_mapCenter())
}
dropPanel.hide()
}
function _mapCenter() {
var centerPoint = Qt.point(editorMap.centerViewport.left + (editorMap.centerViewport.width / 2), editorMap.centerViewport.top + (editorMap.centerViewport.height / 2))
return editorMap.toCoordinate(centerPoint, false /* clipToViewPort */)
}
}
}
}
}
SectionHeader {
id: storageSection
Layout.fillWidth: true
text: qsTr("Storage")
}
GridLayout {
columns: 3
rowSpacing: _margin
columnSpacing: ScreenTools.defaultFontPixelWidth
visible: storageSection.visible
/*QGCButton {
text: qsTr("New...")
Layout.fillWidth: true
onClicked: {
dropPanel.hide()
if (_planMasterController.containsItems) {
mainWindow.showComponentDialog(removeAllPromptDialog, qsTr("New Plan"), mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
}
}
}*/
QGCButton {
text: qsTr("Open...")
Layout.fillWidth: true
enabled: !_planMasterController.syncInProgress
onClicked: {
dropPanel.hide()
if (_planMasterController.dirty) {
mainWindow.showComponentDialog(syncLoadFromFileOverwrite, columnHolder._overwriteText, mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
} else {
_planMasterController.loadFromSelectedFile()
}
}
}
QGCButton {
text: qsTr("Save")
Layout.fillWidth: true
enabled: !_planMasterController.syncInProgress && _planMasterController.currentPlanFile !== ""
onClicked: {
dropPanel.hide()
if(_planMasterController.currentPlanFile !== "") {
_planMasterController.saveToCurrent()
} else {
_planMasterController.saveToSelectedFile()
}
}
}
QGCButton {
text: qsTr("Save As...")
Layout.fillWidth: true
enabled: !_planMasterController.syncInProgress && _planMasterController.containsItems
onClicked: {
dropPanel.hide()
_planMasterController.saveToSelectedFile()
}
}
QGCButton {
Layout.columnSpan: 3
Layout.fillWidth: true
text: qsTr("Save Mission Waypoints As KML...")
enabled: !_planMasterController.syncInProgress && _visualItems.count > 1
onClicked: {
// First point does not count
if (_visualItems.count < 2) {
mainWindow.showComponentDialog(noItemForKML, qsTr("KML"), mainWindow.showDialogDefaultWidth, StandardButton.Cancel)
return
}
dropPanel.hide()
_planMasterController.saveKmlToSelectedFile()
}
}
}
SectionHeader {
id: vehicleSection
Layout.fillWidth: true
text: qsTr("Vehicle")
}
RowLayout {
Layout.fillWidth: true
spacing: _margin
visible: vehicleSection.visible
QGCButton {
text: qsTr("Upload")
Layout.fillWidth: true
enabled: !_planMasterController.offline && !_planMasterController.syncInProgress && _planMasterController.containsItems
visible: !QGroundControl.corePlugin.options.disableVehicleConnection
onClicked: {
dropPanel.hide()
_planMasterController.upload()
}
}
QGCButton {
text: qsTr("Download")
Layout.fillWidth: true
enabled: !_planMasterController.offline && !_planMasterController.syncInProgress
visible: !QGroundControl.corePlugin.options.disableVehicleConnection
onClicked: {
dropPanel.hide()
if (_planMasterController.dirty) {
mainWindow.showComponentDialog(syncLoadFromVehicleOverwrite, columnHolder._overwriteText, mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
} else {
_planMasterController.loadFromVehicle()
}
}
}
QGCButton {
text: qsTr("Clear")
Layout.fillWidth: true
Layout.columnSpan: 2
enabled: !_planMasterController.offline && !_planMasterController.syncInProgress
visible: !QGroundControl.corePlugin.options.disableVehicleConnection
onClicked: {
dropPanel.hide()
mainWindow.showComponentDialog(clearVehicleMissionDialog, text, mainWindow.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
}
}
}
}
}
}