PlanView.qml 33.4 KB
Newer Older
1 2 3 4 5 6 7 8
/****************************************************************************
 *
 *   (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.
 *
 ****************************************************************************/
Don Gagne's avatar
Don Gagne committed
9 10


11 12
import QtQuick          2.3
import QtQuick.Controls 1.2
Don Gagne's avatar
Don Gagne committed
13
import QtQuick.Dialogs  1.2
14 15 16
import QtLocation       5.3
import QtPositioning    5.3
import QtQuick.Layouts  1.2
Don Gagne's avatar
Don Gagne committed
17

18
import QGroundControl               1.0
Don Gagne's avatar
Don Gagne committed
19 20 21
import QGroundControl.FlightMap     1.0
import QGroundControl.ScreenTools   1.0
import QGroundControl.Controls      1.0
22
import QGroundControl.FactSystem    1.0
23
import QGroundControl.FactControls  1.0
Don Gagne's avatar
Don Gagne committed
24
import QGroundControl.Palette       1.0
Don Gagne's avatar
Don Gagne committed
25
import QGroundControl.Mavlink       1.0
26
import QGroundControl.Controllers   1.0
Don Gagne's avatar
Don Gagne committed
27 28

/// Mission Editor
Don Gagne's avatar
Don Gagne committed
29

Don Gagne's avatar
Don Gagne committed
30
QGCView {
31
    id:         _qgcView
32
    viewPanel:  panel
33
    z:          QGroundControl.zOrderTopMost
34

35 36 37 38 39 40 41 42
    readonly property int   _decimalPlaces:             8
    readonly property real  _horizontalMargin:          ScreenTools.defaultFontPixelWidth  / 2
    readonly property real  _margin:                    ScreenTools.defaultFontPixelHeight * 0.5
    readonly property var   _activeVehicle:             QGroundControl.multiVehicleManager.activeVehicle
    readonly property real  _rightPanelWidth:           Math.min(parent.width / 3, ScreenTools.defaultFontPixelWidth * 30)
    readonly property real  _toolButtonTopMargin:       parent.height - ScreenTools.availableHeight + (ScreenTools.defaultFontPixelHeight / 2)
    readonly property var   _defaultVehicleCoordinate:  QtPositioning.coordinate(37.803784, -122.462276)
    readonly property bool  _waypointsOnlyMode:         QGroundControl.corePlugin.options.missionWaypointsOnly
43

44 45 46 47 48
    property var    _planMasterController:      masterController
    property var    _missionController:         _planMasterController.missionController
    property var    _geoFenceController:        _planMasterController.geoFenceController
    property var    _rallyPointController:      _planMasterController.rallyPointController
    property var    _visualItems:               _missionController.visualItems
49 50
    property bool   _lightWidgetBorders:        editorMap.isSatelliteMap
    property bool   _addWaypointOnClick:        false
51
    property bool   _addROIOnClick:             false
52
    property bool   _singleComplexItem:         _missionController.complexMissionItemNames.length === 1
53 54
    property real   _toolbarHeight:             _qgcView.height - ScreenTools.availableHeight
    property int    _editingLayer:              _layerMission
55

56 57 58 59
    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?")
60

61
    Component.onCompleted: {
62
        toolbar.planMasterController =  Qt.binding(function () { return _planMasterController })
63
        toolbar.currentMissionItem =    Qt.binding(function () { return _missionController.currentPlanViewItem })
64 65
    }

66 67 68 69 70
    function addComplexItem(complexItemName) {
        var coordinate = editorMap.center
        coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
        coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
        coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
71
        insertComplexMissionItem(complexItemName, coordinate, _missionController.visualItems.count)
72 73 74
    }

    function insertComplexMissionItem(complexItemName, coordinate, index) {
75
        var sequenceNumber = _missionController.insertComplexMissionItem(complexItemName, coordinate, index)
76
        _missionController.setCurrentPlanViewIndex(sequenceNumber, true)
77 78
    }

79 80 81 82 83
    property bool _firstMissionLoadComplete:    false
    property bool _firstFenceLoadComplete:      false
    property bool _firstRallyLoadComplete:      false
    property bool _firstLoadComplete:           false

84 85 86 87
    MapFitFunctions {
        id:                         mapFitFunctions
        map:                        editorMap
        usePlannedHomePosition:     true
88
        planMasterController:       _planMasterController
89 90
    }

DonLakeFlyer's avatar
DonLakeFlyer committed
91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
    Connections {
        target: QGroundControl.settingsManager.appSettings.defaultMissionItemAltitude

        onRawValueChanged: {
            if (_visualItems.count > 1) {
                _qgcView.showDialog(applyNewAltitude, qsTr("Apply new alititude"), 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()
109
                _missionController.applyDefaultMissionAltitude()
DonLakeFlyer's avatar
DonLakeFlyer committed
110 111 112 113
            }
        }
    }

114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
    Component {
        id: activeMissionUploadDialogComponent

        QGCViewDialog {

            Column {
                anchors.fill:   parent
                spacing:        ScreenTools.defaultFontPixelHeight

                QGCLabel {
                    width:      parent.width
                    wrapMode:   Text.WordWrap
                    text:       qsTr("Your vehicle is currently flying a mission. In order to upload a new or modified mission the current mission will be paused.")
                }

                QGCLabel {
                    width:      parent.width
                    wrapMode:   Text.WordWrap
                    text:       qsTr("After the mission is uploaded you can adjust the current waypoint and start the mission.")
                }

                QGCButton {
                    text:       qsTr("Pause and Upload")
                    onClicked: {
                        _activeVehicle.flightMode = _activeVehicle.pauseFlightMode
139
                        _planMasterController.sendToVehicle()
140
                        hideDialog()
141 142 143 144 145 146
                    }
                }
            }
        }
    }

147 148 149 150 151 152 153 154
    Component {
        id: noItemForKML

        QGCViewMessage {
            message:    qsTr("You need at least one item to create a KML.")
        }
    }

155
    PlanMasterController {
156
        id: masterController
157

158 159
        Component.onCompleted: {
            start(true /* editMode */)
160
            _missionController.setCurrentPlanViewIndex(0, true)
161 162
        }

163
        function upload() {
164
            if (_activeVehicle && _activeVehicle.armed && _activeVehicle.flightMode === _activeVehicle.missionFlightMode) {
165
                _qgcView.showDialog(activeMissionUploadDialogComponent, qsTr("Plan Upload"), _qgcView.showDialogDefaultWidth, StandardButton.Cancel)
166
            } else {
167 168
                sendToVehicle()
            }
DonLakeFlyer's avatar
DonLakeFlyer committed
169 170
        }

171
        function loadFromSelectedFile() {
172
            fileDialog.title =          qsTr("Select Plan File")
173
            fileDialog.selectExisting = true
174
            fileDialog.nameFilters =    masterController.loadNameFilters
175
            fileDialog.openForLoad()
176 177 178
        }

        function saveToSelectedFile() {
179
            fileDialog.title =          qsTr("Save Plan")
180
            fileDialog.plan =           true
181
            fileDialog.selectExisting = false
182
            fileDialog.nameFilters =    masterController.saveNameFilters
183
            fileDialog.openForSave()
184 185
        }

186
        function fitViewportToItems() {
187
            mapFitFunctions.fitMapViewportToMissionItems()
188
        }
189 190 191 192 193 194 195 196

        function saveKmlToSelectedFile() {
            fileDialog.title =          qsTr("Save KML")
            fileDialog.plan =           false
            fileDialog.selectExisting = false
            fileDialog.nameFilters =    masterController.saveKmlFilters
            fileDialog.openForSave()
        }
197
    }
198

199 200
    Connections {
        target: _missionController
201

202
        onNewItemsFromVehicle: {
203 204 205
            if (_visualItems && _visualItems.count != 1) {
                mapFitFunctions.fitMapViewportToMissionItems()
            }
206
            _missionController.setCurrentPlanViewIndex(0, true)
207 208
        }
    }
209

210
    QGCPalette { id: qgcPal; colorGroupEnabled: enabled }
Don Gagne's avatar
Don Gagne committed
211

212 213 214 215
    ExclusiveGroup {
        id: _mapTypeButtonsExclusiveGroup
    }

216 217 218 219
    /// Inserts a new simple mission item
    ///     @param coordinate Location to insert item
    ///     @param index Insert item at this index
    function insertSimpleMissionItem(coordinate, index) {
220
        var sequenceNumber = _missionController.insertSimpleMissionItem(coordinate, index)
221
        _missionController.setCurrentPlanViewIndex(sequenceNumber, true)
222 223
    }

224 225 226 227 228 229 230 231 232 233
    /// Inserts a new ROI mission item
    ///     @param coordinate Location to insert item
    ///     @param index Insert item at this index
    function insertROIMissionItem(coordinate, index) {
        var sequenceNumber = _missionController.insertROIMissionItem(coordinate, index)
        _missionController.setCurrentPlanViewIndex(sequenceNumber, true)
        _addROIOnClick = false
        toolStrip.uncheckAll()
    }

234 235
    property int _moveDialogMissionItemIndex

236 237 238
    QGCFileDialog {
        id:             fileDialog
        qgcView:        _qgcView
239
        property var plan:           true
240
        folder:         QGroundControl.settingsManager.appSettings.missionSavePath
241 242
        fileExtension:  QGroundControl.settingsManager.appSettings.planFileExtension
        fileExtension2: QGroundControl.settingsManager.appSettings.missionFileExtension
243

244
        onAcceptedForSave: {
245
            plan ? masterController.saveToFile(file) : masterController.saveToKml(file)
246
            close()
247 248
        }

249
        onAcceptedForLoad: {
250 251
            masterController.loadFromFile(file)
            masterController.fitViewportToItems()
252
            _missionController.setCurrentPlanViewIndex(0, true)
253
            close()
254 255 256
        }
    }

257 258 259 260 261 262 263 264 265 266
    Component {
        id: moveDialog

        QGCViewDialog {
            function accept() {
                var toIndex = toCombo.currentIndex

                if (toIndex == 0) {
                    toIndex = 1
                }
267
                _missionController.moveMissionItem(_moveDialogMissionItemIndex, toIndex)
268 269 270 271 272 273 274 275 276 277 278 279
                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
280
                    text:           qsTr("Move the selected mission item to the be after following mission item:")
281 282 283 284
                }

                QGCComboBox {
                    id:             toCombo
285
                    model:          _visualItems.count
286 287 288 289 290 291
                    currentIndex:   _moveDialogMissionItemIndex
                }
            }
        }
    }

Don Gagne's avatar
Don Gagne committed
292 293
    QGCViewPanel {
        id:             panel
294
        anchors.fill:   parent
Don Gagne's avatar
Don Gagne committed
295

296
        FlightMap {
297 298 299 300 301
            id:                         editorMap
            anchors.fill:               parent
            mapName:                    "MissionEditor"
            allowGCSLocationCenter:     true
            allowVehicleLocationCenter: true
302
            planView:                   true
Don Gagne's avatar
Don Gagne committed
303

304 305
            // This is the center rectangle of the map which is not obscured by tools
            property rect centerViewport: Qt.rect(_leftToolWidth, _toolbarHeight, editorMap.width - _leftToolWidth - _rightPanelWidth, editorMap.height - _statusHeight - _toolbarHeight)
306

307 308
            property real _leftToolWidth:   toolStrip.x + toolStrip.width
            property real _statusHeight:    waypointValuesDisplay.visible ? editorMap.height - waypointValuesDisplay.y : 0
309

310
            readonly property real animationDuration: 500
311

312 313
            // Initial map position duplicates Fly view position
            Component.onCompleted: editorMap.center = QGroundControl.flightMapPosition
314

315 316 317 318 319 320
            Behavior on zoomLevel {
                NumberAnimation {
                    duration:       editorMap.animationDuration
                    easing.type:    Easing.InOutQuad
                }
            }
321

322 323 324 325 326 327 328 329 330 331 332
            QGCMapPalette { id: mapPal; lightColors: editorMap.isSatelliteMap }

            MouseArea {
                //-- It's a whole lot faster to just fill parent and deal with top offset below
                //   than computing the coordinate offset.
                anchors.fill: parent
                onClicked: {
                    //-- Don't pay attention to items beneath the toolbar.
                    var topLimit = parent.height - ScreenTools.availableHeight
                    if(mouse.y < topLimit) {
                        return
333 334
                    }

335 336 337 338
                    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)
339

340 341 342
                    switch (_editingLayer) {
                    case _layerMission:
                        if (_addWaypointOnClick) {
343
                            insertSimpleMissionItem(coordinate, _missionController.visualItems.count)
344 345 346
                        } else if (_addROIOnClick) {
                            _addROIOnClick = false
                            insertROIMissionItem(coordinate, _missionController.visualItems.count)
347
                        }
348 349
                        break
                    case _layerRallyPoints:
350
                        if (_rallyPointController.supported) {
351
                            _rallyPointController.addPoint(coordinate)
352
                        }
353
                        break
Don Gagne's avatar
Don Gagne committed
354
                    }
Don Gagne's avatar
Don Gagne committed
355
                }
356
            }
Don Gagne's avatar
Don Gagne committed
357

358 359
            // Add the mission item visuals to the map
            Repeater {
360
                model: _editingLayer == _layerMission ? _missionController.visualItems : undefined
361

362 363
                delegate: MissionItemMapVisual {
                    map:        editorMap
DonLakeFlyer's avatar
DonLakeFlyer committed
364
                    qgcView:    _qgcView
365
                    onClicked:  _missionController.setCurrentPlanViewIndex(sequenceNumber, false)
366
                    visible:    _editingLayer == _layerMission
367
                }
368
            }
369

370 371
            // Add lines between waypoints
            MissionLineView {
372
                model: _editingLayer == _layerMission ? _missionController.waypointLines : undefined
373
            }
374

375 376 377 378 379 380 381
            // Add the vehicles to the map
            MapItemView {
                model: QGroundControl.multiVehicleManager.vehicles
                delegate:
                    VehicleMapItem {
                    vehicle:        object
                    coordinate:     object.coordinate
382
                    map:            editorMap
383 384
                    size:           ScreenTools.defaultFontPixelHeight * 3
                    z:              QGroundControl.zOrderMapItems - 1
385
                }
386
            }
387

388 389
            GeoFenceMapVisuals {
                map:                    editorMap
390
                myGeoFenceController:   _geoFenceController
391
                interactive:            _editingLayer == _layerGeoFence
392
                homePosition:           _missionController.plannedHomePosition
393 394
                planView:               true
            }
395

396 397
            RallyPointMapVisuals {
                map:                    editorMap
398
                myRallyPointController: _rallyPointController
399 400
                interactive:            _editingLayer == _layerRallyPoints
                planView:               true
401
            }
402

403 404 405 406 407 408 409 410 411
            ToolStrip {
                id:                 toolStrip
                anchors.leftMargin: ScreenTools.defaultFontPixelWidth
                anchors.left:       parent.left
                anchors.topMargin:  _toolButtonTopMargin
                anchors.top:        parent.top
                color:              qgcPal.window
                title:              qsTr("Plan")
                z:                  QGroundControl.zOrderWidgets
412 413 414 415 416
                showAlternateIcon:  [ false, false, false, masterController.dirty, false, false, false ]
                rotateImage:        [ false, false, false, masterController.syncInProgress, false, false, false ]
                animateImage:       [ false, false, false, masterController.dirty, false, false, false ]
                buttonEnabled:      [ true, true, true, !masterController.syncInProgress, true, true, true ]
                buttonVisible:      [ true, _waypointsOnlyMode, true, true, true, _showZoom, _showZoom ]
417 418 419 420 421 422 423 424 425 426
                maxHeight:          mapScale.y - toolStrip.y

                property bool _showZoom: !ScreenTools.isMobile

                model: [
                    {
                        name:       "Waypoint",
                        iconSource: "/qmlimages/MapAddMission.svg",
                        toggle:     true
                    },
427 428 429 430 431
                    {
                        name:       "ROI",
                        iconSource: "/qmlimages/MapAddMission.svg",
                        toggle:     true
                    },
432
                    {
433
                        name:               _singleComplexItem ? _missionController.complexMissionItemNames[0] : "Pattern",
434 435 436
                        iconSource:         "/qmlimages/MapDrawShape.svg",
                        dropPanelComponent: _singleComplexItem ? undefined : patternDropPanel
                    },
437 438 439 440 441 442
                    {
                        name:                   "Sync",
                        iconSource:             "/qmlimages/MapSync.svg",
                        alternateIconSource:    "/qmlimages/MapSyncChanged.svg",
                        dropPanelComponent:     syncDropPanel
                    },
443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461
                    {
                        name:               "Center",
                        iconSource:         "/qmlimages/MapCenter.svg",
                        dropPanelComponent: centerMapDropPanel
                    },
                    {
                        name:               "In",
                        iconSource:         "/qmlimages/ZoomPlus.svg"
                    },
                    {
                        name:               "Out",
                        iconSource:         "/qmlimages/ZoomMinus.svg"
                    }
                ]

                onClicked: {
                    switch (index) {
                    case 0:
                        _addWaypointOnClick = checked
462
                        _addROIOnClick = false
463 464
                        break
                    case 1:
465 466 467 468
                        _addROIOnClick = checked
                        _addWaypointOnClick = false
                        break
                    case 2:
469
                        if (_singleComplexItem) {
470
                            addComplexItem(_missionController.complexMissionItemNames[0])
471
                        }
472
                        break
473
                    case 5:
474 475
                        editorMap.zoomLevel += 0.5
                        break
476
                    case 6:
477 478 479 480 481 482 483 484 485 486 487 488
                        editorMap.zoomLevel -= 0.5
                        break
                    }
                }
            }
        } // FlightMap

        // Right pane for mission editing controls
        Rectangle {
            id:                 rightPanel
            anchors.bottom:     parent.bottom
            anchors.right:      parent.right
489
            height:             ScreenTools.availableHeight
490 491
            width:              _rightPanelWidth
            color:              qgcPal.window
492 493 494 495 496
            opacity:            0.2
        }

        Item {
            anchors.fill:   rightPanel
497 498 499 500

            // Plan Element selector (Mission/Fence/Rally)
            Row {
                id:                 planElementSelectorRow
501
                anchors.topMargin:  Math.round(ScreenTools.defaultFontPixelHeight / 3)
502 503 504 505
                anchors.top:        parent.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                spacing:            _horizontalMargin
506
                visible:            QGroundControl.corePlugin.options.enablePlanViewSelector
507 508 509 510 511 512 513 514 515

                readonly property real _buttonRadius: ScreenTools.defaultFontPixelHeight * 0.75

                ExclusiveGroup {
                    id: planElementSelectorGroup
                    onCurrentChanged: {
                        switch (current) {
                        case planElementMission:
                            _editingLayer = _layerMission
516
                            break
517 518
                        case planElementGeoFence:
                            _editingLayer = _layerGeoFence
Don Gagne's avatar
Don Gagne committed
519
                            break
520 521
                        case planElementRallyPoints:
                            _editingLayer = _layerRallyPoints
Don Gagne's avatar
Don Gagne committed
522
                            break
523 524 525 526
                        }
                    }
                }

527 528 529 530 531 532 533 534
                QGCRadioButton {
                    id:             planElementMission
                    exclusiveGroup: planElementSelectorGroup
                    text:           qsTr("Mission")
                    checked:        true
                    color:          mapPal.text
                    textStyle:      Text.Outline
                    textStyleColor: mapPal.textOutline
535 536
                }

537 538 539 540 541 542 543 544 545
                Item { height: 1; width: 1 }

                QGCRadioButton {
                    id:             planElementGeoFence
                    exclusiveGroup: planElementSelectorGroup
                    text:           qsTr("Fence")
                    color:          mapPal.text
                    textStyle:      Text.Outline
                    textStyleColor: mapPal.textOutline
546
                }
547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562

                Item { height: 1; width: 1 }

                QGCRadioButton {
                    id:             planElementRallyPoints
                    exclusiveGroup: planElementSelectorGroup
                    text:           qsTr("Rally")
                    color:          mapPal.text
                    textStyle:      Text.Outline
                    textStyleColor: mapPal.textOutline
                }
            } // Row - Plan Element Selector

            // Mission Item Editor
            Item {
                id:                 missionItemEditor
563
                anchors.topMargin:  ScreenTools.defaultFontPixelHeight / 2
564 565 566 567 568 569 570 571 572 573 574
                anchors.top:        planElementSelectorRow.visible ? planElementSelectorRow.bottom : planElementSelectorRow.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                anchors.bottom:     parent.bottom
                visible:            _editingLayer == _layerMission

                QGCListView {
                    id:             missionItemEditorListView
                    anchors.fill:   parent
                    spacing:        _margin / 2
                    orientation:    ListView.Vertical
575
                    model:          _missionController.visualItems
576 577
                    cacheBuffer:    Math.max(height * 2, 0)
                    clip:           true
578
                    currentIndex:   _missionController.currentPlanViewIndex
579 580 581
                    highlightMoveDuration: 250

                    delegate: MissionItemEditor {
582
                        map:                editorMap
583
                        masterController:  _planMasterController
584 585 586 587
                        missionItem:        object
                        width:              parent.width
                        readOnly:           false
                        rootQgcView:        _qgcView
588

589
                        onClicked:  _missionController.setCurrentPlanViewIndex(object.sequenceNumber, false)
590 591 592

                        onRemove: {
                            var removeIndex = index
593 594
                            _missionController.removeMissionItem(removeIndex)
                            if (removeIndex >= _missionController.visualItems.count) {
595 596
                                removeIndex--
                            }
597
                            _missionController.setCurrentPlanViewIndex(removeIndex, true)
598 599
                        }

600 601
                        onInsertWaypoint:       insertSimpleMissionItem(editorMap.center, index)
                        onInsertComplexItem:    insertComplexMissionItem(complexItemName, editorMap.center, index)
602 603 604 605 606
                    }
                } // QGCListView
            } // Item - Mission Item editor

            // GeoFence Editor
607 608 609 610 611 612
            GeoFenceEditor {
                anchors.topMargin:      ScreenTools.defaultFontPixelHeight / 2
                anchors.top:            planElementSelectorRow.bottom
                anchors.left:           parent.left
                anchors.right:          parent.right
                availableHeight:        ScreenTools.availableHeight
613
                myGeoFenceController:   _geoFenceController
614 615
                flightMap:              editorMap
                visible:                _editingLayer == _layerGeoFence
616 617 618 619 620 621
            }

            // Rally Point Editor

            RallyPointEditorHeader {
                id:                 rallyPointHeader
622 623
                anchors.topMargin:  ScreenTools.defaultFontPixelHeight / 2
                anchors.top:        planElementSelectorRow.bottom
624 625 626
                anchors.left:       parent.left
                anchors.right:      parent.right
                visible:            _editingLayer == _layerRallyPoints
627
                controller:         _rallyPointController
628 629 630 631
            }

            RallyPointItemEditor {
                id:                 rallyPointEditor
632 633
                anchors.topMargin:  ScreenTools.defaultFontPixelHeight / 2
                anchors.top:        rallyPointHeader.bottom
634 635
                anchors.left:       parent.left
                anchors.right:      parent.right
636 637 638
                visible:            _editingLayer == _layerRallyPoints && _rallyPointController.points.count
                rallyPoint:         _rallyPointController.currentRallyPoint
                controller:         _rallyPointController
639 640
            }
        } // Right panel
641 642 643 644 645 646 647 648 649 650 651 652 653 654

        MapScale {
            id:                 mapScale
            anchors.margins:    ScreenTools.defaultFontPixelHeight * (0.66)
            anchors.bottom:     waypointValuesDisplay.visible ? waypointValuesDisplay.top : parent.bottom
            anchors.left:       parent.left
            mapControl:         editorMap
            visible:            !ScreenTools.isTinyScreen
        }

        MissionItemStatus {
            id:                 waypointValuesDisplay
            anchors.margins:    ScreenTools.defaultFontPixelWidth
            anchors.left:       parent.left
655
            maxWidth:           parent.width - rightPanel.width - x
656
            anchors.bottom:     parent.bottom
657
            missionItems:       _missionController.visualItems
658
            visible:            _editingLayer === _layerMission && !ScreenTools.isShortScreen
659
        }
Don Gagne's avatar
Don Gagne committed
660
    } // QGCViewPanel
661

662 663 664 665
    Component {
        id: syncLoadFromVehicleOverwrite
        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
Don Gagne's avatar
Don Gagne committed
666
            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?")
667 668
            function accept() {
                hideDialog()
669
                masterController.loadFromVehicle()
670 671 672 673 674 675 676 677
            }
        }
    }

    Component {
        id: syncLoadFromFileOverwrite
        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
DonLakeFlyer's avatar
DonLakeFlyer committed
678
            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?")
679 680
            function accept() {
                hideDialog()
681
                masterController.loadFromSelectedFile()
682 683 684 685
            }
        }
    }

686 687 688
    Component {
        id: removeAllPromptDialog
        QGCViewMessage {
689 690
            message: qsTr("Are you sure you want to remove all items? ") +
                     (_planMasterController.offline ? "" : qsTr("This will also remove all items from the vehicle."))
691
            function accept() {
692 693 694 695 696
                if (_planMasterController.offline) {
                    masterController.removeAll()
                } else {
                    masterController.removeAllFromVehicle()
                }
697 698 699 700 701
                hideDialog()
            }
        }
    }

702 703 704 705 706 707 708 709 710 711 712
    //- ToolStrip DropPanel Components

    Component {
        id: centerMapDropPanel

        CenterMapDropPanel {
            map:            editorMap
            fitFunctions:   mapFitFunctions
        }
    }

713 714 715 716 717 718 719 720 721
    Component {
        id: patternDropPanel

        ColumnLayout {
            spacing:    ScreenTools.defaultFontPixelWidth * 0.5

            QGCLabel { text: qsTr("Create complex pattern:") }

            Repeater {
722
                model: _missionController.complexMissionItemNames
723 724 725 726 727 728

                QGCButton {
                    text:               modelData
                    Layout.fillWidth:   true

                    onClicked: {
729
                        addComplexItem(modelData)
730 731 732 733 734 735
                        dropPanel.hide()
                    }
                }
            }
        } // Column
    }
736 737

    Component {
738
        id: syncDropPanel
739

740 741 742
        Column {
            id:         columnHolder
            spacing:    _margin
743

744
            property string _overwriteText: (_editingLayer == _layerMission) ? qsTr("Mission overwrite") : ((_editingLayer == _layerGeoFence) ? qsTr("GeoFence overwrite") : qsTr("Rally Points overwrite"))
745

746 747 748
            QGCLabel {
                width:      sendSaveGrid.width
                wrapMode:   Text.WordWrap
749
                text:       masterController.dirty ?
750 751
                                qsTr("You have unsaved changes. You should upload to your vehicle, or save to a file:") :
                                qsTr("Sync:")
752 753
            }

754 755 756 757 758 759
            GridLayout {
                id:                 sendSaveGrid
                columns:            2
                anchors.margins:    _margin
                rowSpacing:         _margin
                columnSpacing:      ScreenTools.defaultFontPixelWidth
760

761 762 763
                QGCButton {
                    text:               qsTr("Upload")
                    Layout.fillWidth:   true
764
                    enabled:            !masterController.offline && !masterController.syncInProgress
765 766
                    onClicked: {
                        dropPanel.hide()
767
                        masterController.upload()
768 769
                    }
                }
770

771 772 773
                QGCButton {
                    text:               qsTr("Download")
                    Layout.fillWidth:   true
774
                    enabled:            !masterController.offline && !masterController.syncInProgress
775 776
                    onClicked: {
                        dropPanel.hide()
777
                        if (masterController.dirty) {
778 779
                            _qgcView.showDialog(syncLoadFromVehicleOverwrite, columnHolder._overwriteText, _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
                        } else {
780
                            masterController.loadFromVehicle()
781 782 783
                        }
                    }
                }
784

785 786 787
                QGCButton {
                    text:               qsTr("Save To File...")
                    Layout.fillWidth:   true
788
                    enabled:            !masterController.syncInProgress
789 790
                    onClicked: {
                        dropPanel.hide()
791
                        masterController.saveToSelectedFile()
792 793 794 795 796 797
                    }
                }

                QGCButton {
                    text:               qsTr("Load From File...")
                    Layout.fillWidth:   true
798
                    enabled:            !masterController.syncInProgress
799 800
                    onClicked: {
                        dropPanel.hide()
801
                        if (masterController.dirty) {
802 803
                            _qgcView.showDialog(syncLoadFromFileOverwrite, columnHolder._overwriteText, _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
                        } else {
804
                            masterController.loadFromSelectedFile()
805 806 807 808 809 810 811 812 813 814 815 816
                        }
                    }
                }

                QGCButton {
                    text:               qsTr("Remove All")
                    Layout.fillWidth:   true
                    onClicked:  {
                        dropPanel.hide()
                        _qgcView.showDialog(removeAllPromptDialog, qsTr("Remove all"), _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
                    }
                }
817 818 819 820 821 822

                QGCButton {
                    text:               qsTr("Save KML...")
                    Layout.fillWidth:   true
                    enabled:            !masterController.syncInProgress
                    onClicked: {
823 824 825 826 827
                        // First point do not count
                        if (_visualItems.count < 2) {
                            _qgcView.showDialog(noItemForKML, qsTr("KML"), _qgcView.showDialogDefaultWidth, StandardButton.Cancel)
                            return
                        }
828 829 830 831
                        dropPanel.hide()
                        masterController.saveKmlToSelectedFile()
                    }
                }
832
            }
833 834
        }
    }
Don Gagne's avatar
Don Gagne committed
835
} // QGCVIew