PlanView.qml 35.8 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 22
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
23
import QGroundControl.Mavlink       1.0
24
import QGroundControl.Controllers   1.0
Don Gagne's avatar
Don Gagne committed
25 26

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

Don Gagne's avatar
Don Gagne committed
28
QGCView {
29
    id:         _qgcView
30
    viewPanel:  panel
Don Gagne's avatar
Don Gagne committed
31

32
    // zOrder comes from the Loader in MainWindow.qml
Gus Grubba's avatar
Gus Grubba committed
33
    z: QGroundControl.zOrderTopMost
34

35 36 37 38 39
    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)
Gus Grubba's avatar
Gus Grubba committed
40
    readonly property real      _rightPanelOpacity:     1
41 42
    readonly property int       _toolButtonCount:       6
    readonly property real      _toolButtonTopMargin:   parent.height - ScreenTools.availableHeight + (ScreenTools.defaultFontPixelHeight / 2)
43
    readonly property var       _defaultVehicleCoordinate:   QtPositioning.coordinate(37.803784, -122.462276)
44

45
    property var    _visualItems:           missionController.visualItems
Don Gagne's avatar
Don Gagne committed
46
    property var    _currentMissionItem
47
    property int    _currentMissionIndex:   0
48
    property bool   _lightWidgetBorders:    editorMap.isSatelliteMap
49
    property bool   _addWaypointOnClick:    false
50
    property bool   _singleComplexItem:     missionController.complexMissionItemNames.length === 1
51
    property real   _toolbarHeight:   _qgcView.height - ScreenTools.availableHeight
52

53 54 55 56 57
    /// The controller which should be called for load/save, send to/from vehicle calls
    property var _syncDropDownController: missionController

    readonly property int _layerMission:        1
    readonly property int _layerGeoFence:       2
58
    readonly property int _layerRallyPoints:    3
59 60
    property int _editingLayer: _layerMission

61 62 63 64 65
    Component.onCompleted: {
        toolbar.missionController =     Qt.binding(function () { return missionController })
        toolbar.currentMissionItem =    Qt.binding(function () { return _currentMissionItem })
    }

66 67 68 69 70 71
    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)
        var sequenceNumber = missionController.insertComplexMissionItem(complexItemName, coordinate, missionController.visualItems.count)
Donald Gagne's avatar
Donald Gagne committed
72
        setCurrentItem(sequenceNumber)
73 74
    }

75 76 77 78 79 80 81 82
    property bool _firstMissionLoadComplete:    false
    property bool _firstFenceLoadComplete:      false
    property bool _firstRallyLoadComplete:      false
    property bool _firstLoadComplete:           false

    function checkFirstLoadComplete() {
        if (!_firstLoadComplete && _firstMissionLoadComplete && _firstRallyLoadComplete && _firstFenceLoadComplete) {
            _firstLoadComplete = true
83
            mapFitFunctions.fitMapViewportToAllItems()
84 85 86
        }
    }

87 88 89
    MapFitFunctions {
        id:                         mapFitFunctions
        map:                        editorMap
90
        mapFitViewport:             editorMap.centerViewport
91 92 93 94 95 96
        usePlannedHomePosition:     true
        mapGeoFenceController:      geoFenceController
        mapMissionController:       missionController
        mapRallyPointController:    rallyPointController
    }

97
    MissionController {
98
        id: missionController
99

100 101
        property var nameFilters: [ qsTr("Mission Files (*.%1)").arg(missionController.fileExtension) , qsTr("All Files (*.*)") ]

102 103
        Component.onCompleted: {
            start(true /* editMode */)
104
            setCurrentItem(0)
105 106
        }

DonLakeFlyer's avatar
DonLakeFlyer committed
107 108 109 110 111
        // Users is switching away from Plan View
        function saveOnSwitch() {
            save()
        }

112
        function loadFromSelectedFile() {
113 114 115 116
            fileDialog.title =          qsTr("Select Mission File")
            fileDialog.selectExisting = true
            fileDialog.nameFilters =    missionController.nameFilters
            fileDialog.openForLoad()
117 118 119
        }

        function saveToSelectedFile() {
120 121 122 123
            fileDialog.title =          qsTr("Save Mission")
            fileDialog.selectExisting = false
            fileDialog.nameFilters =    missionController.nameFilters
            fileDialog.openForSave()
124 125
        }

126
        function fitViewportToItems() {
127
            mapFitFunctions.fitMapViewportToMissionItems()
128 129
        }

130
        onVisualItemsChanged: itemDragger.clearItem()
131

132
        onNewItemsFromVehicle: {
133 134 135
            if (_visualItems && _visualItems.count != 1) {
                mapFitFunctions.fitMapViewportToMissionItems()
            }
136
            setCurrentItem(0)
137 138
            _firstMissionLoadComplete = true
            checkFirstLoadComplete()
139 140
        }
    }
141

142 143
    GeoFenceController {
        id: geoFenceController
144

145 146
        property var nameFilters: [ qsTr("GeoFence Files (*.%1)").arg(geoFenceController.fileExtension) , qsTr("All Files (*.*)") ]

147
        Component.onCompleted: start(true /* editMode */)
148

149
        function saveToSelectedFile() {
150 151 152 153
            fileDialog.title =          qsTr("Save GeoFence")
            fileDialog.selectExisting = false
            fileDialog.nameFilters =    geoFenceController.nameFilters
            fileDialog.openForSave()
154 155 156
        }

        function loadFromSelectedFile() {
157 158 159 160 161
            fileDialog.title =          qsTr("Select GeoFence File")
            fileDialog.selectExisting = true
            fileDialog.nameFilters =    geoFenceController.nameFilters
            fileDialog.openForLoad()
            ///mapFitFunctions.fitMapViewportToFenceItems()
162 163
        }

164
        function fitViewportToItems() {
165
            mapFitFunctions.fitMapViewportToFenceItems()
166 167 168 169 170 171
        }

        onLoadComplete: {
            _firstFenceLoadComplete = true
            switch (_syncDropDownController) {
            case geoFenceController:
172
                mapFitFunctions.fitMapViewportToFenceItems()
173 174 175 176 177 178
                break
            case missionController:
                checkFirstLoadComplete()
                break
            }
        }
179
    }
180

181 182 183
    RallyPointController {
        id: rallyPointController

184 185
        property var nameFilters: [ qsTr("Rally Point Files (*.%1)").arg(rallyPointController.fileExtension) , qsTr("All Files (*.*)") ]

186 187 188 189 190 191 192 193 194 195 196
        onCurrentRallyPointChanged: {
            if (_editingLayer == _layerRallyPoints && !currentRallyPoint) {
                itemDragger.visible = false
                itemDragger.coordinateItem = undefined
                itemDragger.mapCoordinateIndicator = undefined
            }
        }

        Component.onCompleted: start(true /* editMode */)

        function saveToSelectedFile() {
197 198 199 200
            fileDialog.title =          qsTr("Save Rally Points")
            fileDialog.selectExisting = false
            fileDialog.nameFilters =    rallyPointController.nameFilters
            fileDialog.openForSave()
201 202 203
        }

        function loadFromSelectedFile() {
204 205 206 207 208
            fileDialog.title =          qsTr("Select Rally Point File")
            fileDialog.selectExisting = true
            fileDialog.nameFilters =    rallyPointController.nameFilters
            fileDialog.openForLoad()
            //mapFitFunctions.fitMapViewportToRallyItems()
209 210 211
        }

        function fitViewportToItems() {
212
            mapFitFunctions.fitMapViewportToRallyItems()
213 214 215 216 217 218
        }

        onLoadComplete: {
            _firstRallyLoadComplete = true
            switch (_syncDropDownController) {
            case rallyPointController:
219
                mapFitFunctions.fitMapViewportToRallyItems()
220 221 222 223
                break
            case missionController:
                checkFirstLoadComplete()
                break
224 225 226 227
            }
        }
    }

228
    QGCPalette { id: qgcPal; colorGroupEnabled: enabled }
Don Gagne's avatar
Don Gagne committed
229

230 231 232 233
    ExclusiveGroup {
        id: _mapTypeButtonsExclusiveGroup
    }

234 235
    /// Sets a new current mission item
    ///     @param sequenceNumber - index for new item, -1 to clear current item
236
    function setCurrentItem(sequenceNumber) {
237 238 239 240 241 242 243 244
        if (sequenceNumber !== _currentMissionIndex) {
            _currentMissionItem = undefined
            _currentMissionIndex = -1
            for (var i=0; i<_visualItems.count; i++) {
                var visualItem = _visualItems.get(i)
                if (visualItem.sequenceNumber == sequenceNumber) {
                    _currentMissionItem = visualItem
                    _currentMissionItem.isCurrentItem = true
245
                    _currentMissionIndex = sequenceNumber
246 247 248
                } else {
                    visualItem.isCurrentItem = false
                }
Don Gagne's avatar
Don Gagne committed
249
            }
250 251 252
        }
    }

253 254 255 256 257 258 259 260 261
    /// Inserts a new simple mission item
    ///     @param coordinate Location to insert item
    ///     @param index Insert item at this index
    function insertSimpleMissionItem(coordinate, index) {
        setCurrentItem(-1)
        var sequenceNumber = missionController.insertSimpleMissionItem(coordinate, index)
        setCurrentItem(sequenceNumber)
    }

262 263
    property int _moveDialogMissionItemIndex

264 265 266 267 268
    QGCFileDialog {
        id:             fileDialog
        qgcView:        _qgcView
        folder:         QGroundControl.settingsManager.appSettings.missionSavePath
        fileExtension:  _syncDropDownController.fileExtension
269

270 271 272
        onAcceptedForSave: {
            _syncDropDownController.saveToFile(file)
            close()
273 274
        }

275 276 277 278 279
        onAcceptedForLoad: {
            _syncDropDownController.loadFromFile(file)
            _syncDropDownController.fitViewportToItems()
            _currentMissionItem = _visualItems.get(0)
            close()
280 281 282
        }
    }

283 284 285 286 287 288 289 290 291 292
    Component {
        id: moveDialog

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

                if (toIndex == 0) {
                    toIndex = 1
                }
293
                missionController.moveMissionItem(_moveDialogMissionItemIndex, toIndex)
294 295 296 297 298 299 300 301 302 303 304 305
                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
306
                    text:           qsTr("Move the selected mission item to the be after following mission item:")
307 308 309 310
                }

                QGCComboBox {
                    id:             toCombo
311
                    model:          _visualItems.count
312 313 314 315 316 317
                    currentIndex:   _moveDialogMissionItemIndex
                }
            }
        }
    }

Don Gagne's avatar
Don Gagne committed
318 319
    QGCViewPanel {
        id:             panel
320 321 322 323
        height:         ScreenTools.availableHeight
        anchors.bottom: parent.bottom
        anchors.left:   parent.left
        anchors.right:  parent.right
Don Gagne's avatar
Don Gagne committed
324

325 326 327 328 329 330 331
        FlightMap {
            id:             editorMap
            height:         _qgcView.height
            anchors.bottom: parent.bottom
            anchors.left:   parent.left
            anchors.right:  parent.right//rightPanel.left
            mapName:        "MissionEditor"
Don Gagne's avatar
Don Gagne committed
332

333 334
            // 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)
335

336 337
            property real _leftToolWidth:   toolStrip.x + toolStrip.width
            property real _statusHeight:    waypointValuesDisplay.visible ? editorMap.height - waypointValuesDisplay.y : 0
338

339
            readonly property real animationDuration: 500
340

341 342
            // Initial map position duplicates Fly view position
            Component.onCompleted: editorMap.center = QGroundControl.flightMapPosition
343

344 345 346 347 348 349
            Behavior on zoomLevel {
                NumberAnimation {
                    duration:       editorMap.animationDuration
                    easing.type:    Easing.InOutQuad
                }
            }
350

351 352 353 354 355 356 357 358 359 360 361
            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
362 363
                    }

364 365 366 367
                    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)
368

369 370 371 372
                    switch (_editingLayer) {
                    case _layerMission:
                        if (_addWaypointOnClick) {
                            insertSimpleMissionItem(coordinate, missionController.visualItems.count)
373
                        }
374 375 376 377
                        break
                    case _layerRallyPoints:
                        if (rallyPointController.rallyPointsSupported) {
                            rallyPointController.addPoint(coordinate)
378
                        }
379
                        break
Don Gagne's avatar
Don Gagne committed
380
                    }
Don Gagne's avatar
Don Gagne committed
381
                }
382
            }
Don Gagne's avatar
Don Gagne committed
383

384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409
            // We use this item to support dragging since dragging a MapQuickItem just doesn't seem to work
            Rectangle {
                id:             itemDragger
                x:              mapCoordinateIndicator ? (mapCoordinateIndicator.x + mapCoordinateIndicator.anchorPoint.x - (itemDragger.width / 2)) : 100
                y:              mapCoordinateIndicator ? (mapCoordinateIndicator.y + mapCoordinateIndicator.anchorPoint.y - (itemDragger.height / 2)) : 100
                width:          ScreenTools.defaultFontPixelHeight * 3
                height:         ScreenTools.defaultFontPixelHeight * 3
                color:          "transparent"
                visible:        false
                z:              QGroundControl.zOrderMapItems + 1    // Above item icons

                property var    coordinateItem
                property var    mapCoordinateIndicator
                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, false /* clipToViewPort */)
                        coordinate.altitude = itemDragger.coordinateItem.coordinate.altitude
                        itemDragger.preventCoordinateBindingLoop = true
                        itemDragger.coordinateItem.coordinate = coordinate
                        itemDragger.preventCoordinateBindingLoop = false
Don Gagne's avatar
Don Gagne committed
410
                    }
411
                }
412

413 414 415 416
                function clearItem() {
                    itemDragger.visible = false
                    itemDragger.coordinateItem = undefined
                    itemDragger.mapCoordinateIndicator = undefined
417 418
                }

419 420 421
                Drag.active:    itemDrag.drag.active
                Drag.hotSpot.x: width  / 2
                Drag.hotSpot.y: height / 2
Don Gagne's avatar
Don Gagne committed
422

423 424 425 426 427 428 429 430
                MouseArea {
                    id:             itemDrag
                    anchors.fill:   parent
                    drag.target:    parent
                    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
431
                }
432
            }
Don Gagne's avatar
Don Gagne committed
433

434 435 436
            // Add the mission item visuals to the map
            Repeater {
                model: missionController.visualItems
437

438 439 440
                delegate: MissionItemMapVisual {
                    map:        editorMap
                    onClicked:  setCurrentItem(sequenceNumber)
441
                }
442
            }
443

444 445 446 447
            // Add lines between waypoints
            MissionLineView {
                model:      _editingLayer == _layerMission ? missionController.waypointLines : undefined
            }
448

449 450 451 452 453 454 455 456 457 458
            // Add the vehicles to the map
            MapItemView {
                model: QGroundControl.multiVehicleManager.vehicles
                delegate:
                    VehicleMapItem {
                    vehicle:        object
                    coordinate:     object.coordinate
                    isSatellite:    editorMap.isSatelliteMap
                    size:           ScreenTools.defaultFontPixelHeight * 3
                    z:              QGroundControl.zOrderMapItems - 1
459
                }
460 461 462 463 464 465 466 467
            }
            GeoFenceMapVisuals {
                map:                    editorMap
                myGeoFenceController:   geoFenceController
                interactive:            _editingLayer == _layerGeoFence
                homePosition:           missionController.plannedHomePosition
                planView:               true
            }
468

469
            // Rally points on map
470

471 472
            MapItemView {
                model: rallyPointController.points
473

474 475 476 477 478 479
                delegate: MapQuickItem {
                    id:             itemIndicator
                    anchorPoint.x:  sourceItem.anchorPointX
                    anchorPoint.y:  sourceItem.anchorPointY
                    coordinate:     object.coordinate
                    z:              QGroundControl.zOrderMapItems
480

481 482 483 484
                    sourceItem: MissionItemIndexLabel {
                        id:         itemIndexLabel
                        label:      qsTr("R", "rally point map item label")
                        checked:    _editingLayer == _layerRallyPoints ? object == rallyPointController.currentRallyPoint : false
485

486
                        onClicked: rallyPointController.currentRallyPoint = object
487

488 489 490 491 492 493
                        onCheckedChanged: {
                            if (checked) {
                                // Setup our drag item
                                itemDragger.visible = true
                                itemDragger.coordinateItem = Qt.binding(function() { return object })
                                itemDragger.mapCoordinateIndicator = Qt.binding(function() { return itemIndicator })
494 495 496 497
                            }
                        }
                    }
                }
498
            }
499

500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550
            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
                buttonEnabled:      [ true, true, true, true, true ]
                buttonVisible:      [ true, true, true, _showZoom, _showZoom ]
                maxHeight:          mapScale.y - toolStrip.y

                property bool _showZoom: !ScreenTools.isMobile

                property bool mySingleComplexItem: _singleComplexItem

                model: [
                    {
                        name:       "Waypoint",
                        iconSource: "/qmlimages/MapAddMission.svg",
                        toggle:     true
                    },
                    {
                        name:               "Pattern",
                        iconSource:         "/qmlimages/MapDrawShape.svg",
                        dropPanelComponent: _singleComplexItem ? undefined : patternDropPanel
                    },
                    {
                        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
                        break
                    case 1:
                        if (_singleComplexItem) {
                            addComplexItem(missionController.complexMissionItemNames[0])
551
                        }
552 553 554 555 556 557 558 559 560 561
                        break
                    case 3:
                        editorMap.zoomLevel += 0.5
                        break
                    case 4:
                        editorMap.zoomLevel -= 0.5
                        break
                    }
                }
            }
562

563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615
            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
                anchors.bottom:         parent.bottom
                z:                      QGroundControl.zOrderTopMost
                currentMissionItem:     _currentMissionItem
                missionItems:           missionController.visualItems
                expandedWidth:          missionItemEditor.x - (ScreenTools.defaultFontPixelWidth * 2)
                missionDistance:        missionController.missionDistance
                missionTime:            missionController.missionTime
                missionMaxTelemetry:    missionController.missionMaxTelemetry
                visible:                _editingLayer == _layerMission && !ScreenTools.isShortScreen
            }
        } // FlightMap

        // Right pane for mission editing controls
        Rectangle {
            id:                 rightPanel
            anchors.top:        parent.top
            anchors.bottom:     parent.bottom
            anchors.right:      parent.right
            width:              _rightPanelWidth
            color:              qgcPal.window
            opacity:            0.95

            // Plan Element selector (Mission/Fence/Rally)
            Row {
                id:                 planElementSelectorRow
                anchors.top:        parent.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                spacing:            _horizontalMargin
                visible:            false // WIP: Temporarily remove - QGroundControl.corePlugin.options.enablePlanViewSelector

                readonly property real _buttonRadius: ScreenTools.defaultFontPixelHeight * 0.75

                ExclusiveGroup {
                    id: planElementSelectorGroup
                    onCurrentChanged: {
                        switch (current) {
                        case planElementMission:
                            _editingLayer = _layerMission
                            _syncDropDownController = missionController
616
                            break
617 618 619
                        case planElementGeoFence:
                            _editingLayer = _layerGeoFence
                            _syncDropDownController = geoFenceController
Don Gagne's avatar
Don Gagne committed
620
                            break
621 622 623
                        case planElementRallyPoints:
                            _editingLayer = _layerRallyPoints
                            _syncDropDownController = rallyPointController
Don Gagne's avatar
Don Gagne committed
624
                            break
625
                        }
626
                        _syncDropDownController.fitViewportToItems()
627 628 629
                    }
                }

630 631 632 633 634 635 636 637
                QGCRadioButton {
                    id:             planElementMission
                    exclusiveGroup: planElementSelectorGroup
                    text:           qsTr("Mission")
                    checked:        true
                    color:          mapPal.text
                    textStyle:      Text.Outline
                    textStyleColor: mapPal.textOutline
638 639
                }

640 641 642 643 644 645 646 647 648
                Item { height: 1; width: 1 }

                QGCRadioButton {
                    id:             planElementGeoFence
                    exclusiveGroup: planElementSelectorGroup
                    text:           qsTr("Fence")
                    color:          mapPal.text
                    textStyle:      Text.Outline
                    textStyleColor: mapPal.textOutline
649
                }
650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739

                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
                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
                    model:          missionController.visualItems
                    cacheBuffer:    Math.max(height * 2, 0)
                    clip:           true
                    currentIndex:   _currentMissionIndex
                    highlightMoveDuration: 250

                    delegate: MissionItemEditor {
                        map:            editorMap
                        missionItem:    object
                        width:          parent.width
                        readOnly:       false
                        rootQgcView:    _qgcView

                        onClicked:  setCurrentItem(object.sequenceNumber)

                        onRemove: {
                            var removeIndex = index
                            itemDragger.clearItem()
                            missionController.removeMissionItem(removeIndex)
                            if (removeIndex >= missionController.visualItems.count) {
                                removeIndex--
                            }
                            setCurrentItem(removeIndex)
                        }

                        onInsert: insertSimpleMissionItem(editorMap.center, index)
                    }
                } // QGCListView
            } // Item - Mission Item editor

            // GeoFence Editor
            Loader {
                anchors.top:        planElementSelectorRow.visible ? planElementSelectorRow.bottom : planElementSelectorRow.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                sourceComponent:    _editingLayer == _layerGeoFence ? geoFenceEditorComponent : undefined

                property real   availableWidth:         _rightPanelWidth
                property real   availableHeight:        ScreenTools.availableHeight
                property var    myGeoFenceController:   geoFenceController
            }

            // Rally Point Editor

            RallyPointEditorHeader {
                id:                 rallyPointHeader
                anchors.top:        planElementSelectorRow.visible ? planElementSelectorRow.bottom : planElementSelectorRow.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                visible:            _editingLayer == _layerRallyPoints
                controller:         rallyPointController
            }

            RallyPointItemEditor {
                id:                 rallyPointEditor
                anchors.top:        planElementSelectorRow.visible ? planElementSelectorRow.bottom : planElementSelectorRow.top
                anchors.left:       parent.left
                anchors.right:      parent.right
                visible:            _editingLayer == _layerRallyPoints && rallyPointController.points.count
                rallyPoint:         rallyPointController.currentRallyPoint
                controller:         rallyPointController
            }
        } // Right panel
Don Gagne's avatar
Don Gagne committed
740
    } // QGCViewPanel
741

742 743 744 745
    Component {
        id: syncLoadFromVehicleOverwrite
        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
Don Gagne's avatar
Don Gagne committed
746
            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?")
747 748
            function accept() {
                hideDialog()
Don Gagne's avatar
Don Gagne committed
749
                _syncDropDownController.loadFromVehicle()
750 751 752 753 754 755 756 757
            }
        }
    }

    Component {
        id: syncLoadFromFileOverwrite
        QGCViewMessage {
            id:         syncLoadFromVehicleCheck
Don Gagne's avatar
Don Gagne committed
758
            message:   qsTr("You have unsaved/unsent changes. Loading a from a file will lose these changes. Are you sure you want to load from a file?")
759 760
            function accept() {
                hideDialog()
Don Gagne's avatar
Don Gagne committed
761
                _syncDropDownController.loadFromSelectedFile()
762 763 764 765
            }
        }
    }

766 767 768
    Component {
        id: removeAllPromptDialog
        QGCViewMessage {
Don Gagne's avatar
Don Gagne committed
769
            message: qsTr("Are you sure you want to remove all items?")
770 771
            function accept() {
                itemDragger.clearItem()
Don Gagne's avatar
Don Gagne committed
772
                _syncDropDownController.removeAll()
773 774 775 776 777
                hideDialog()
            }
        }
    }

778 779
    //- ToolStrip DropPanel Components

780
    Component {
781
        id: syncDropPanel
782

783 784 785
        Column {
            id:         columnHolder
            spacing:    _margin
786

787
            property string _overwriteText: (_editingLayer == _layerMission) ? qsTr("Mission overwrite") : ((_editingLayer == _layerGeoFence) ? qsTr("GeoFence overwrite") : qsTr("Rally Points overwrite"))
Don Gagne's avatar
Don Gagne committed
788

789
            QGCLabel {
dogmaphobic's avatar
dogmaphobic committed
790
                width:      sendSaveGrid.width
791
                wrapMode:   Text.WordWrap
792
                text:       _syncDropDownController.dirty ?
Don Gagne's avatar
Don Gagne committed
793
                                qsTr("You have unsaved changes. You should send to your vehicle, or save to a file:") :
794
                                qsTr("Sync:")
795
            }
796

dogmaphobic's avatar
dogmaphobic committed
797 798 799 800 801 802
            GridLayout {
                id:                 sendSaveGrid
                columns:            2
                anchors.margins:    _margin
                rowSpacing:         _margin
                columnSpacing:      ScreenTools.defaultFontPixelWidth
803

804
                QGCButton {
DonLakeFlyer's avatar
DonLakeFlyer committed
805
                    text:               qsTr("Save")
dogmaphobic's avatar
dogmaphobic committed
806
                    Layout.fillWidth:   true
DonLakeFlyer's avatar
DonLakeFlyer committed
807
                    enabled:            !_syncDropDownController.syncInProgress
808
                    onClicked: {
809
                        dropPanel.hide()
DonLakeFlyer's avatar
DonLakeFlyer committed
810
                        _syncDropDownController.save()
811 812
                    }
                }
813

814
                QGCButton {
dogmaphobic's avatar
dogmaphobic committed
815 816
                    text:               qsTr("Load From Vehicle")
                    Layout.fillWidth:   true
817
                    enabled:            _activeVehicle && !_syncDropDownController.syncInProgress
818
                    onClicked: {
819
                        dropPanel.hide()
820
                        if (_syncDropDownController.dirty) {
821
                            _qgcView.showDialog(syncLoadFromVehicleOverwrite, columnHolder._overwriteText, _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
822
                        } else {
823
                            _syncDropDownController.loadFromVehicle()
824
                        }
825 826
                    }
                }
827

828
                QGCButton {
dogmaphobic's avatar
dogmaphobic committed
829 830
                    text:               qsTr("Save To File...")
                    Layout.fillWidth:   true
831
                    enabled:            !_syncDropDownController.syncInProgress
832
                    onClicked: {
833
                        dropPanel.hide()
834
                        _syncDropDownController.saveToSelectedFile()
835 836
                    }
                }
837

838
                QGCButton {
dogmaphobic's avatar
dogmaphobic committed
839 840
                    text:               qsTr("Load From File...")
                    Layout.fillWidth:   true
841
                    enabled:            !_syncDropDownController.syncInProgress
842
                    onClicked: {
843
                        dropPanel.hide()
844
                        if (_syncDropDownController.dirty) {
845
                            _qgcView.showDialog(syncLoadFromFileOverwrite, columnHolder._overwriteText, _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.Cancel)
846
                        } else {
847
                            _syncDropDownController.loadFromSelectedFile()
848
                        }
849 850
                    }
                }
851

dogmaphobic's avatar
dogmaphobic committed
852 853 854 855
                QGCButton {
                    text:               qsTr("Remove All")
                    Layout.fillWidth:   true
                    onClicked:  {
856
                        dropPanel.hide()
857
                        _qgcView.showDialog(removeAllPromptDialog, qsTr("Remove all"), _qgcView.showDialogDefaultWidth, StandardButton.Yes | StandardButton.No)
dogmaphobic's avatar
dogmaphobic committed
858
                    }
859 860
                }
            }
861 862
        }
    }
863 864 865 866 867 868 869 870 871 872

    Component {
        id: centerMapDropPanel

        CenterMapDropPanel {
            map:            editorMap
            fitFunctions:   mapFitFunctions
        }
    }

873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888
    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: {
889
                        addComplexItem(modelData)
890 891 892 893 894 895
                        dropPanel.hide()
                    }
                }
            }
        } // Column
    }
896 897 898 899 900 901 902 903 904 905 906

    Component {
        id: geoFenceEditorComponent

        GeoFenceEditor {
            availableWidth:         _rightPanelWidth
            availableHeight:        ScreenTools.availableHeight
            myGeoFenceController:   geoFenceController
            flightMap:              editorMap
        }
    }
Don Gagne's avatar
Don Gagne committed
907
} // QGCVIew