MissionEditor.qml 32.8 KB
Newer Older
Don Gagne's avatar
Don Gagne committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009, 2015 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>

This file is part of the QGROUNDCONTROL project

    QGROUNDCONTROL is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    QGROUNDCONTROL is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with QGROUNDCONTROL. If not, see <http://www.gnu.org/licenses/>.

======================================================================*/

Don Gagne's avatar
Don Gagne committed
24 25 26 27 28
import QtQuick          2.4
import QtQuick.Controls 1.3
import QtQuick.Dialogs  1.2
import QtLocation       5.3
import QtPositioning    5.3
Don Gagne's avatar
Don Gagne committed
29

30
import QGroundControl               1.0
Don Gagne's avatar
Don Gagne committed
31 32 33 34
import QGroundControl.FlightMap     1.0
import QGroundControl.ScreenTools   1.0
import QGroundControl.Controls      1.0
import QGroundControl.Palette       1.0
Don Gagne's avatar
Don Gagne committed
35
import QGroundControl.Mavlink       1.0
36
import QGroundControl.Controllers   1.0
Don Gagne's avatar
Don Gagne committed
37 38

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

Don Gagne's avatar
Don Gagne committed
40
QGCView {
41 42
    id:         _root

43
    property bool syncNeeded: controller.visualItems.dirty // Unsaved changes, visible to parent container
44

45 46
    viewPanel:          panel
    topDialogMargin:    height - mainWindow.availableHeight
Don Gagne's avatar
Don Gagne committed
47

48
    // zOrder comes from the Loader in MainWindow.qml
Gus Grubba's avatar
Gus Grubba committed
49
    z: QGroundControl.zOrderTopMost
50

51
    readonly property int       _decimalPlaces:     8
52
    readonly property real      _horizontalMargin:  ScreenTools.defaultFontPixelWidth  / 2
53 54 55
    readonly property real      _margin:            ScreenTools.defaultFontPixelHeight / 2
    readonly property var       _activeVehicle:     multiVehicleManager.activeVehicle
    readonly property real      _editFieldWidth:    ScreenTools.defaultFontPixelWidth * 16
Don Gagne's avatar
Don Gagne committed
56
    readonly property real      _rightPanelWidth:   Math.min(parent.width / 3, ScreenTools.defaultFontPixelWidth * 30)
57 58 59 60
    readonly property real      _rightPanelOpacity: 0.8
    readonly property int       _toolButtonCount:   6
    readonly property string    _autoSyncKey:       "AutoSync"
    readonly property int       _addMissionItemsButtonAutoOffTimeout:   10000
61
    readonly property var       _defaultVehicleCoordinate:   QtPositioning.coordinate(37.803784, -122.462276)
62

63
    property var    _visualItems:          controller.visualItems
Don Gagne's avatar
Don Gagne committed
64
    property var    _currentMissionItem
65 66
    property bool   _firstVehiclePosition:  true
    property var    activeVehiclePosition:  _activeVehicle ? _activeVehicle.coordinate : QtPositioning.coordinate()
67

68
    onActiveVehiclePositionChanged: updateMapToVehiclePosition()
69

70 71 72 73 74 75 76 77
    Connections {
        target: multiVehicleManager

        onActiveVehicleChanged: {
            // When the active vehicle changes we need to allow the first vehicle position to move the map again
            _firstVehiclePosition = true
            updateMapToVehiclePosition()
        }
78
    }
79 80

    function updateMapToVehiclePosition() {
81
        if (_activeVehicle && _activeVehicle.coordinateValid && _activeVehicle.coordinate.isValid && _firstVehiclePosition) {
82 83
            _firstVehiclePosition = false
            editorMap.center = _activeVehicle.coordinate
84 85 86
        }
    }

87 88 89 90 91
    function loadFromVehicle() {
        controller.getMissionItems()
    }

    function loadFromFile() {
92 93 94
        if (ScreenTools.isMobile) {
            _root.showDialog(mobileFilePicker, "Select Mission File", _root.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
        } else {
Don Gagne's avatar
Don Gagne committed
95
            controller.loadMissionFromFilePicker()
96 97 98 99 100 101 102 103
            fitViewportToMissionItems()
        }
    }

    function saveToFile() {
        if (ScreenTools.isMobile) {
            _root.showDialog(mobileFileSaver, "Save Mission File", _root.showDialogDefaultWidth, StandardButton.Save | StandardButton.Cancel)
        } else {
Don Gagne's avatar
Don Gagne committed
104
            controller.saveMissionToFile()
105
        }
106 107 108 109 110 111 112 113 114 115 116 117
    }

    function normalizeLat(lat) {
        // Normalize latitude to range: 0 to 180, S to N
        return lat + 90.0
    }

    function normalizeLon(lon) {
        // Normalize longitude to range: 0 to 360, W to E
        return lon  + 180.0
    }

118
    /// Fix the map viewport to the current mission items.
119
    function fitViewportToMissionItems() {
120 121
        if (_visualItems.count == 1) {
            editorMap.center = _visualItems.get(0).coordinate
122
        } else {
123
            var missionItem = _visualItems.get(0)
124 125 126 127 128
            var north = normalizeLat(missionItem.coordinate.latitude)
            var south = north
            var east = normalizeLon(missionItem.coordinate.longitude)
            var west = east

129 130
            for (var i=1; i<_visualItems.count; i++) {
                missionItem = _visualItems.get(i)
131

132
                if (missionItem.specifiesCoordinate && !missionItem.isStandaloneCoordinate) {
133 134 135 136 137 138 139 140
                    var lat = normalizeLat(missionItem.coordinate.latitude)
                    var lon = normalizeLon(missionItem.coordinate.longitude)

                    north = Math.max(north, lat)
                    south = Math.min(south, lat)
                    east = Math.max(east, lon)
                    west = Math.min(west, lon)
                }
141
            }
142
            editorMap.visibleRegion = QtPositioning.rectangle(QtPositioning.coordinate(north - 90.0, west - 180.0), QtPositioning.coordinate(south - 90.0, east - 180.0))
143 144 145
        }
    }

146
    MissionController {
147
        id:         controller
148

149 150
        Component.onCompleted: {
            start(true /* editMode */)
151
            setCurrentItem(0)
152 153
        }

Don Gagne's avatar
Don Gagne committed
154
        /*
155 156
        FIXME: autoSync is temporarily disconnected since it's still buggy

157 158 159
        autoSync:   QGroundControl.flightMapSettings.loadMapSetting(editorMap.mapName, _autoSyncKey, true)

        onAutoSyncChanged:      QGroundControl.flightMapSettings.saveMapSetting(editorMap.mapName, _autoSyncKey, autoSync)
160
*/
161

162
        onVisualItemsChanged: itemDragger.clearItem()
163
        onNewItemsFromVehicle: fitViewportToMissionItems()
164
    }
165

166
    QGCPalette { id: qgcPal; colorGroupEnabled: enabled }
Don Gagne's avatar
Don Gagne committed
167

168 169 170 171 172 173
    ExclusiveGroup {
        id: _mapTypeButtonsExclusiveGroup
    }

    ExclusiveGroup {
        id: _dropButtonsExclusiveGroup
174 175
    }

176
    function setCurrentItem(sequenceNumber) {
Don Gagne's avatar
Don Gagne committed
177
        _currentMissionItem = undefined
178
        for (var i=0; i<_visualItems.count; i++) {
179 180 181
            var visualItem = _visualItems.get(i)
            if (visualItem.sequenceNumber == sequenceNumber) {
                _currentMissionItem = visualItem
Don Gagne's avatar
Don Gagne committed
182 183
                _currentMissionItem.isCurrentItem = true
            } else {
184
                visualItem.isCurrentItem = false
Don Gagne's avatar
Don Gagne committed
185
            }
186 187 188
        }
    }

189 190
    property int _moveDialogMissionItemIndex

191 192 193
    Component {
        id: mobileFilePicker

Don Gagne's avatar
Don Gagne committed
194 195 196 197 198 199 200
        QGCMobileFileDialog {
            openDialog:     true
            fileExtension:  QGroundControl.missionFileExtension

            onFilenameReturned: {
                controller.loadMissionFromFile(filename)
                fitViewportToMissionItems()
201 202 203 204 205 206 207
            }
        }
    }

    Component {
        id: mobileFileSaver

Don Gagne's avatar
Don Gagne committed
208 209 210
        QGCMobileFileDialog {
            openDialog:     false
            fileExtension:  QGroundControl.missionFileExtension
211

Don Gagne's avatar
Don Gagne committed
212 213
            onFilenameReturned: {
                controller.saveMissionToFile(filename)
214 215 216 217
            }
        }
    }

218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
    Component {
        id: moveDialog

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

                if (toIndex == 0) {
                    toIndex = 1
                }
                controller.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:           "Move the selected mission item to the be after following mission item:"
                }

                QGCComboBox {
                    id:             toCombo
246
                    model:          _visualItems.count
247 248 249 250 251 252
                    currentIndex:   _moveDialogMissionItemIndex
                }
            }
        }
    }

Don Gagne's avatar
Don Gagne committed
253 254 255
    QGCViewPanel {
        id:             panel
        anchors.fill:   parent
Don Gagne's avatar
Don Gagne committed
256

Don Gagne's avatar
Don Gagne committed
257
        Item {
Don Gagne's avatar
Don Gagne committed
258 259
            anchors.fill: parent

Don Gagne's avatar
Don Gagne committed
260 261
            FlightMap {
                id:             editorMap
262
                anchors.fill:   parent
Don Gagne's avatar
Don Gagne committed
263
                mapName:        "MissionEditor"
264

265 266
                signal mapClicked(var coordinate)

267 268
                readonly property real animationDuration: 500

269 270 271
                // Initial map position duplicates Fly view position
                Component.onCompleted: editorMap.center = QGroundControl.flightMapPosition

272 273 274 275 276 277 278
                Behavior on zoomLevel {
                    NumberAnimation {
                        duration:       editorMap.animationDuration
                        easing.type:    Easing.InOutQuad
                    }
                }

Don Gagne's avatar
Don Gagne committed
279 280 281 282
                MouseArea {
                    anchors.fill: parent

                    onClicked: {
283 284 285 286
                        var coordinate = editorMap.toCoordinate(Qt.point(mouse.x, mouse.y))
                        coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
                        coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
                        coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
287
                        if (addMissionItemsButton.checked) {
288 289
                            var sequenceNumber = controller.insertSimpleMissionItem(coordinate, controller.visualItems.count)
                            setCurrentItem(sequenceNumber)
290
                        } else {
291
                            editorMap.mapClicked(coordinate)
292
                        }
Don Gagne's avatar
Don Gagne committed
293
                    }
Don Gagne's avatar
Don Gagne committed
294
                }
Don Gagne's avatar
Don Gagne committed
295

296
                // We use this item to support dragging since dragging a MapQuickItem just doesn't seem to work
Don Gagne's avatar
Don Gagne committed
297 298 299 300
                Rectangle {
                    id:             itemDragger
                    x:              missionItemIndicator ? (missionItemIndicator.x + missionItemIndicator.anchorPoint.x - (itemDragger.width / 2)) : 100
                    y:              missionItemIndicator ? (missionItemIndicator.y + missionItemIndicator.anchorPoint.y - (itemDragger.height / 2)) : 100
301 302
                    width:          ScreenTools.defaultFontPixelHeight * 2
                    height:         ScreenTools.defaultFontPixelHeight * 2
Don Gagne's avatar
Don Gagne committed
303 304 305
                    color:          "transparent"
                    visible:        false
                    z:              QGroundControl.zOrderMapItems + 1    // Above item icons
306 307 308

                    property var    missionItem
                    property var    missionItemIndicator
309 310 311 312 313 314 315 316 317 318 319 320 321 322 323
                    property bool   preventCoordinateBindingLoop: false

                    onXChanged: liveDrag()
                    onYChanged: liveDrag()

                    function liveDrag() {
                        if (!itemDragger.preventCoordinateBindingLoop && Drag.active) {
                            var point = Qt.point(itemDragger.x + (itemDragger.width  / 2), itemDragger.y + (itemDragger.height / 2))
                            var coordinate = editorMap.toCoordinate(point)
                            coordinate.altitude = itemDragger.missionItem.coordinate.altitude
                            itemDragger.preventCoordinateBindingLoop = true
                            itemDragger.missionItem.coordinate = coordinate
                            itemDragger.preventCoordinateBindingLoop = false
                        }
                    }
Don Gagne's avatar
Don Gagne committed
324

325
                    function clearItem() {
Don Gagne's avatar
Don Gagne committed
326 327 328 329 330
                        itemDragger.visible = false
                        itemDragger.missionItem = undefined
                        itemDragger.missionItemIndicator = undefined
                    }

331 332 333 334 335 336 337 338
                    Drag.active:    itemDrag.drag.active
                    Drag.hotSpot.x: width  / 2
                    Drag.hotSpot.y: height / 2

                    MouseArea {
                        id:             itemDrag
                        anchors.fill:   parent
                        drag.target:    parent
Don Gagne's avatar
Don Gagne committed
339 340 341 342
                        drag.minimumX:  0
                        drag.minimumY:  0
                        drag.maximumX:  itemDragger.parent.width - parent.width
                        drag.maximumY:  itemDragger.parent.height - parent.height
Don Gagne's avatar
Don Gagne committed
343
                    }
344
                }
345

346
                // Add the complex mission item polygon to the map
347
                MapItemView {
Don Gagne's avatar
Don Gagne committed
348 349
                    model: controller.complexVisualItems
                    delegate: MapPolygon {
350 351 352 353 354 355
                        color:      'green'
                        path:       object.polygonPath
                        opacity:    0.5
                    }
                }

356 357 358 359 360 361 362 363 364 365
                // Add the complex mission item grid to the map
                MapItemView {
                    model: controller.complexVisualItems

                    delegate: MapPolyline {
                        line.color: "white"
                        path:       object.gridPoints
                    }
                }

366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381
                // Add the complex mission item exit coordinates
                MapItemView {
                    model:      controller.complexVisualItems
                    delegate:   exitCoordinateComponent
                }

                Component {
                    id: exitCoordinateComponent

                    MissionItemIndicator {
                        coordinate:     object.exitCoordinate
                        z:              QGroundControl.zOrderMapItems
                        missionItem:    object
                        sequenceNumber: object.lastSequenceNumber
                        visible:        object.specifiesCoordinate
                    }
Don Gagne's avatar
Don Gagne committed
382 383
                }

384 385 386 387 388 389
                // Add the simple mission items to the map
                MapItemView {
                    model:      controller.visualItems
                    delegate:   missionItemComponent
                }

Don Gagne's avatar
Don Gagne committed
390
                Component {
391
                    id: missionItemComponent
Don Gagne's avatar
Don Gagne committed
392 393 394 395

                    MissionItemIndicator {
                        id:             itemIndicator
                        coordinate:     object.coordinate
396
                        visible:        object.specifiesCoordinate
Don Gagne's avatar
Don Gagne committed
397
                        z:              QGroundControl.zOrderMapItems
398
                        missionItem:    object
399
                        sequenceNumber: object.sequenceNumber
Don Gagne's avatar
Don Gagne committed
400 401 402

                        onClicked: setCurrentItem(object.sequenceNumber)

403 404
                        function updateItemIndicator() {
                            if (object.isCurrentItem && itemIndicator.visible && object.specifiesCoordinate && object.isSimpleItem) {
405 406 407 408
                                // Setup our drag item
                                itemDragger.visible = true
                                itemDragger.missionItem = Qt.binding(function() { return object })
                                itemDragger.missionItemIndicator = Qt.binding(function() { return itemIndicator })
Don Gagne's avatar
Don Gagne committed
409 410 411
                            }
                        }

412 413 414
                        Connections {
                            target: object

415 416
                            onIsCurrentItemChanged:         updateItemIndicator()
                            onSpecifiesCoordinateChanged:   updateItemIndicator()
417 418
                        }

Don Gagne's avatar
Don Gagne committed
419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436
                        // These are the non-coordinate child mission items attached to this item
                        Row {
                            anchors.top:    parent.top
                            anchors.left:   parent.right

                            Repeater {
                                model: object.childItems

                                delegate: MissionItemIndexLabel {
                                    label:          object.sequenceNumber
                                    isCurrentItem:  object.isCurrentItem
                                    z:              2

                                    onClicked: setCurrentItem(object.sequenceNumber)
                                }
                            }
                        }
                    }
437 438 439
                }

                // Add lines between waypoints
440 441
                MissionLineView {
                    model:          controller.waypointLines
Don Gagne's avatar
Don Gagne committed
442 443
                }

Don Gagne's avatar
Don Gagne committed
444 445 446 447 448 449 450 451 452 453 454 455 456
                // Add the vehicles to the map
                MapItemView {
                    model: multiVehicleManager.vehicles
                    delegate:
                        VehicleMapItem {
                                vehicle:        object
                                coordinate:     object.coordinate
                                isSatellite:    editorMap.isSatelliteMap
                                size:           ScreenTools.defaultFontPixelHeight * 5
                                z:              QGroundControl.zOrderMapItems - 1
                        }
                }

457
                // Mission Item Editor
Don Gagne's avatar
Don Gagne committed
458
                Item {
459
                    id:             missionItemEditor
460
                    height:         mainWindow.availableHeight
461 462 463 464
                    anchors.bottom: parent.bottom
                    anchors.right:  parent.right
                    width:          _rightPanelWidth
                    opacity:        _rightPanelOpacity
Gus Grubba's avatar
Gus Grubba committed
465
                    z:              QGroundControl.zOrderTopMost
466 467

                    ListView {
468 469 470 471
                        id:             editorListView
                        anchors.left:   parent.left
                        anchors.right:  parent.right
                        anchors.top:    parent.top
472
                        height:         parent.height
473 474
                        spacing:        _margin / 2
                        orientation:    ListView.Vertical
475
                        model:          controller.visualItems
476
                        cacheBuffer:    height * 2
477

478
                        delegate: MissionItemEditor {
479 480
                            missionItem:    object
                            width:          parent.width
Don Gagne's avatar
Don Gagne committed
481
                            qgcView:        _root
482
                            readOnly:       false
483 484 485 486

                            onClicked:  setCurrentItem(object.sequenceNumber)

                            onRemove: {
487
                                itemDragger.clearItem()
488
                                controller.removeMissionItem(object.sequenceNumber)
489 490
                            }

491
                            onInsert: {
492 493
                                var sequenceNumber = controller.insertSimpleMissionItem(editorMap.center, i)
                                setCurrentItem(sequenceNumber)
494
                            }
495

496
                            onMoveHomeToMapCenter: controller.visualItems.get(0).coordinate = editorMap.center
497
                        }
498 499 500
                    } // ListView
                } // Item - Mission Item editor

501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516
                //-- Dismiss Drop Down (if any)
                MouseArea {
                    anchors.fill:   parent
                    enabled:        _dropButtonsExclusiveGroup.current != null
                    onClicked: {
                        if(_dropButtonsExclusiveGroup.current)
                            _dropButtonsExclusiveGroup.current.checked = false
                        _dropButtonsExclusiveGroup.current = null
                    }
                }

                Item {
                    id:     toolbarSpacer
                    height: mainWindow.tbHeight
                    width:  1
                }
517

518 519 520 521 522 523 524 525 526 527 528 529
                //-- Vertical Tool Buttons
                Column {
                    id:                         toolColumn
                    anchors.margins:            ScreenTools.defaultFontPixelHeight
                    anchors.left:               parent.left
                    anchors.top:                toolbarSpacer.bottom
                    spacing:                    ScreenTools.defaultFontPixelHeight

                    RoundButton {
                        id:                 addMissionItemsButton
                        buttonImage:        "/qmlimages/MapAddMission.svg"
                        z:                  QGroundControl.zOrderWidgets
530 531
                    }

532 533 534 535 536 537 538 539 540 541 542
                    RoundButton {
                        id:                 addShapeButton
                        buttonImage:        "/qmlimages/MapDrawShape.svg"
                        z:                  QGroundControl.zOrderWidgets
                        visible:            QGroundControl.experimentalSurvey

                        onClicked: {
                            var coordinate = editorMap.center
                            coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
                            coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
                            coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
543 544
                            var sequenceNumber = controller.insertComplexMissionItem(coordinate, controller.visualItems.count)
                            setCurrentItem(sequenceNumber)
545
                            checked = false
546
                            addMissionItemsButton.checked = false
547 548 549
                        }
                    }

550 551 552
                    DropButton {
                        id:                 syncButton
                        dropDirection:      dropRight
553
                        buttonImage:        syncNeeded ? "/qmlimages/MapSyncChanged.svg" : "/qmlimages/MapSync.svg"
554 555 556 557
                        viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                        exclusiveGroup:     _dropButtonsExclusiveGroup
                        z:                  QGroundControl.zOrderWidgets
                        dropDownComponent:  syncDropDownComponent
558 559
                        enabled:            !controller.syncInProgress
                        rotateImage:        controller.syncInProgress
560 561
                    }

562 563 564 565 566 567 568
                    DropButton {
                        id:                 centerMapButton
                        dropDirection:      dropRight
                        buttonImage:        "/qmlimages/MapCenter.svg"
                        viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                        exclusiveGroup:     _dropButtonsExclusiveGroup
                        z:                  QGroundControl.zOrderWidgets
569

570 571 572
                        dropDownComponent: Component {
                            Column {
                                QGCLabel { text: "Center map:" }
573

574 575
                                Row {
                                    spacing: ScreenTools.defaultFontPixelWidth
576

577
                                    QGCButton {
578
                                        text: "Home"
579 580 581

                                        onClicked: {
                                            centerMapButton.hideDropDown()
582
                                            editorMap.center = controller.visualItems.get(0).coordinate
583
                                        }
584 585
                                    }

586 587 588 589 590 591 592
                                    QGCButton {
                                        text: "Mission"

                                        onClicked: {
                                            centerMapButton.hideDropDown()
                                            fitViewportToMissionItems()
                                        }
593 594
                                    }

595 596 597
                                    QGCButton {
                                        text:       "Vehicle"
                                        enabled:    activeVehicle && activeVehicle.latitude != 0 && activeVehicle.longitude != 0
598

599
                                        property var activeVehicle: multiVehicleManager.activeVehicle
600

601 602
                                        onClicked: {
                                            centerMapButton.hideDropDown()
603
                                            editorMap.center = activeVehicle.coordinate
604
                                        }
605 606 607 608 609 610
                                    }
                                }
                            }
                        }
                    }

611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639
                    DropButton {
                        id:                 mapTypeButton
                        dropDirection:      dropRight
                        buttonImage:        "/qmlimages/MapType.svg"
                        viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                        exclusiveGroup:     _dropButtonsExclusiveGroup
                        z:                  QGroundControl.zOrderWidgets

                        dropDownComponent: Component {
                            Column {
                                QGCLabel { text: "Map type:" }

                                Row {
                                    spacing: ScreenTools.defaultFontPixelWidth

                                    Repeater {
                                        model: QGroundControl.flightMapSettings.mapTypes

                                        QGCButton {
                                            checkable:      true
                                            checked:        editorMap.mapType == text
                                            text:           modelData
                                            exclusiveGroup: _mapTypeButtonsExclusiveGroup

                                            onClicked: {
                                                editorMap.mapType = text
                                                checked = true
                                                mapTypeButton.hideDropDown()
                                            }
640 641 642 643 644 645 646
                                        }
                                    }
                                }
                            }
                        }
                    }

647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671
                    //-- Zoom Map In
                    RoundButton {
                        id:                 mapZoomPlus
                        visible:            !ScreenTools.isTinyScreen && !ScreenTools.isShortScreen
                        buttonImage:        "/qmlimages/ZoomPlus.svg"
                        z:                  QGroundControl.zOrderWidgets
                        onClicked: {
                            if(editorMap)
                                editorMap.zoomLevel += 0.5
                            checked = false
                        }
                    }

                    //-- Zoom Map Out
                    RoundButton {
                        id:                 mapZoomMinus
                        visible:            !ScreenTools.isTinyScreen && !ScreenTools.isShortScreen
                        buttonImage:        "/qmlimages/ZoomMinus.svg"
                        z:                  QGroundControl.zOrderWidgets
                        onClicked: {
                            if(editorMap)
                                editorMap.zoomLevel -= 0.5
                            checked = false
                        }
                    }
672
                }
673

674
                MissionItemStatus {
675
                    id:                 waypointValuesDisplay
676
                    anchors.margins:    ScreenTools.defaultFontPixelWidth
677 678
                    anchors.left:       parent.left
                    anchors.bottom:     parent.bottom
679 680
                    z:                  QGroundControl.zOrderTopMost
                    currentMissionItem: _currentMissionItem
681
                    missionItems:       controller.visualItems
682
                    expandedWidth:      missionItemEditor.x - (ScreenTools.defaultFontPixelWidth * 2)
Don Gagne's avatar
Don Gagne committed
683
                    visible:            !ScreenTools.isShortScreen
684
                }
685
            } // FlightMap
Don Gagne's avatar
Don Gagne committed
686 687
        } // Item - split view container
    } // QGCViewPanel
688

689 690 691 692 693
    Component {
        id: syncLoadFromVehicleOverwrite

        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
694
            message:   "You have unsaved/unsent mission changes. Loading the mission from the Vehicle will lose these changes. Are you sure you want to load the mission from the Vehicle?"
695 696 697

            function accept() {
                hideDialog()
698
                loadFromVehicle()
699 700 701 702 703 704 705 706 707
            }
        }
    }

    Component {
        id: syncLoadFromFileOverwrite

        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
708
            message:   "You have unsaved/unsent mission changes. Loading a mission from a file will lose these changes. Are you sure you want to load a mission from a file?"
709 710 711

            function accept() {
                hideDialog()
712
                loadFromFile()
713 714 715 716
            }
        }
    }

717 718 719 720 721 722 723 724 725 726 727 728 729 730
    Component {
        id: removeAllPromptDialog

        QGCViewMessage {
            message: "Are you sure you want to delete all mission items?"

            function accept() {
                itemDragger.clearItem()
                controller.removeAllMissionItems()
                hideDialog()
            }
        }
    }

731 732 733 734 735 736 737 738
    Component {
        id: syncDropDownComponent

        Column {
            id:         columnHolder
            spacing:    _margin

            QGCLabel {
Don Gagne's avatar
Don Gagne committed
739
                width:      sendSaveRow.width
740
                wrapMode:   Text.WordWrap
741
                text:       syncNeeded && !controller.autoSync ?
742 743 744 745 746
                                "You have unsaved changed to you mission. You should send to your vehicle, or save to a file:" :
                                "Sync:"
            }

            Row {
Don Gagne's avatar
Don Gagne committed
747
                id:         sendSaveRow
748
                visible:    true //autoSyncCheckBox.enabled && autoSyncCheckBox.checked
749 750 751 752
                spacing:    ScreenTools.defaultFontPixelWidth

                QGCButton {
                    text:       "Send to vehicle"
753
                    enabled:    _activeVehicle && !controller.syncInProgress
754 755 756 757 758 759 760 761 762

                    onClicked: {
                        syncButton.hideDropDown()
                        controller.sendMissionItems()
                    }
                }

                QGCButton {
                    text:       "Load from vehicle"
763
                    enabled:    _activeVehicle && !controller.syncInProgress
764 765 766

                    onClicked: {
                        syncButton.hideDropDown()
767 768 769
                        if (syncNeeded) {
                            _root.showDialog(syncLoadFromVehicleOverwrite, "Mission overwrite", _root.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
                        } else {
770
                            loadFromVehicle()
771
                        }
772 773 774 775 776 777 778 779 780
                    }
                }
            }

            Row {
                spacing: ScreenTools.defaultFontPixelWidth

                QGCButton {
                    text:       "Save to file..."
781
                    enabled:    !controller.syncInProgress
782 783 784

                    onClicked: {
                        syncButton.hideDropDown()
785
                        saveToFile()
786 787 788 789 790
                    }
                }

                QGCButton {
                    text:       "Load from file..."
791
                    enabled:    !controller.syncInProgress
792 793 794

                    onClicked: {
                        syncButton.hideDropDown()
795 796 797
                        if (syncNeeded) {
                            _root.showDialog(syncLoadFromFileOverwrite, "Mission overwrite", _root.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
                        } else {
798
                            loadFromFile()
799
                        }
800 801 802
                    }
                }
            }
803 804 805 806 807 808 809 810 811

            QGCButton {
                text:       "Remove all"
                onClicked:  {
                    syncButton.hideDropDown()
                    _root.showDialog(removeAllPromptDialog, "Delete all", _root.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
                }
            }

812 813
/*
        FIXME: autoSync is temporarily disconnected since it's still buggy
814 815 816 817 818 819 820 821 822 823 824 825 826 827 828

            QGCLabel {
                id:         autoSyncDisallowedLabel
                visible:    _activeVehicle && _activeVehicle.armed
                text:       "AutoSync is not allowed whie vehicle is armed"
            }

            QGCCheckBox {
                id:         autoSyncCheckBox
                checked:    controller.autoSync
                text:       "Automatically sync changes with vehicle"
                enabled:    _activeVehicle ? !_activeVehicle.armed : false

                onClicked: controller.autoSync = checked
            }
829
*/
830 831
        }
    }
Don Gagne's avatar
Don Gagne committed
832
} // QGCVIew