MissionEditor.qml 51 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 41
QGCView {
    viewPanel: panel
Don Gagne's avatar
Don Gagne committed
42

43
    // zOrder comes from the Loader in MainWindow.qml
Gus Grubba's avatar
Gus Grubba committed
44
    z: QGroundControl.zOrderTopMost
45

46 47 48 49 50 51 52 53 54
    readonly property int       _decimalPlaces:     8
    readonly property real      _horizontalMargin:  ScreenTools.defaultFontPixelWidth / 2
    readonly property real      _margin:            ScreenTools.defaultFontPixelHeight / 2
    readonly property var       _activeVehicle:     multiVehicleManager.activeVehicle
    readonly property real      _editFieldWidth:    ScreenTools.defaultFontPixelWidth * 16
    readonly property real      _rightPanelWidth:   ScreenTools.defaultFontPixelWidth * 30
    readonly property real      _rightPanelOpacity: 0.8
    readonly property int       _toolButtonCount:   6
    readonly property string    _autoSyncKey:       "AutoSync"
Don Gagne's avatar
Don Gagne committed
55
    readonly property string    _showHelpKey:       "ShowHelp"
56
    readonly property int       _addMissionItemsButtonAutoOffTimeout:   10000
57
    readonly property var       _defaultVehicleCoordinate:   QtPositioning.coordinate(37.803784, -122.462276)
58

59 60 61 62
    property var    _missionItems:  controller.missionItems

    property bool   gpsLock:        _activeVehicle ? _activeVehicle.coordinateValid : false
    property bool   _firstGpsLock:  true
63

64 65
    //property var    _homePositionManager:       QGroundControl.homePositionManager
    //property string _homePositionName:          _homePositionManager.homePositions.get(0).name
66
    //property var    offlineHomePosition:        _homePositionManager.homePositions.get(0).coordinate
67

68 69
    property var    liveHomePosition:           controller.liveHomePosition
    property var    liveHomePositionAvailable:  controller.liveHomePositionAvailable
70
    property var    homePosition:               _defaultVehicleCoordinate
71

72 73
    property bool _syncNeeded:                  controller.missionItems.dirty
    property bool _syncInProgress:              _activeVehicle ? _activeVehicle.missionManager.inProgress : false
74

Don Gagne's avatar
Don Gagne committed
75 76
    property bool _showHelp:                    QGroundControl.flightMapSettings.loadBoolMapSetting(editorMap.mapName, _showHelpKey, true)

77 78 79 80 81 82 83 84 85 86 87
    onGpsLockChanged:       updateMapToVehiclePosition()
    Component.onCompleted:  updateMapToVehiclePosition()

    function updateMapToVehiclePosition() {
        if (gpsLock && _firstGpsLock) {
            _firstGpsLock = false
            editorMap.latitude = _activeVehicle.latitude
            editorMap.longitude = _activeVehicle.longitude
        }
    }

88
    MissionController {
89
        id:         controller
90 91

        Component.onCompleted: start(true /* editMode */)
Don Gagne's avatar
Don Gagne committed
92
        /*
93 94
        FIXME: autoSync is temporarily disconnected since it's still buggy

95 96 97
        autoSync:   QGroundControl.flightMapSettings.loadMapSetting(editorMap.mapName, _autoSyncKey, true)

        onAutoSyncChanged:      QGroundControl.flightMapSettings.saveMapSetting(editorMap.mapName, _autoSyncKey, autoSync)
98
*/
99

Don Gagne's avatar
Don Gagne committed
100
        onMissionItemsChanged: itemDragger.clearItem()
101
    }
102

103
    QGCPalette { id: qgcPal; colorGroupEnabled: enabled }
Don Gagne's avatar
Don Gagne committed
104

105 106 107 108 109 110
    ExclusiveGroup {
        id: _mapTypeButtonsExclusiveGroup
    }

    ExclusiveGroup {
        id: _dropButtonsExclusiveGroup
111 112
    }

113 114 115 116 117 118
    function setCurrentItem(index) {
        for (var i=0; i<_missionItems.count; i++) {
            _missionItems.get(i).isCurrentItem = (i == index)
        }
    }

Don Gagne's avatar
Don Gagne committed
119 120 121
    QGCViewPanel {
        id:             panel
        anchors.fill:   parent
Don Gagne's avatar
Don Gagne committed
122

Don Gagne's avatar
Don Gagne committed
123
        Item {
Don Gagne's avatar
Don Gagne committed
124 125
            anchors.fill: parent

Don Gagne's avatar
Don Gagne committed
126 127
            FlightMap {
                id:             editorMap
128
                anchors.fill:   parent
Don Gagne's avatar
Don Gagne committed
129
                mapName:        "MissionEditor"
130 131
                latitude:       tabletPosition.latitude
                longitude:      tabletPosition.longitude
Don Gagne's avatar
Don Gagne committed
132

133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155
                readonly property real animationDuration: 500

                Behavior on zoomLevel {
                    NumberAnimation {
                        duration:       editorMap.animationDuration
                        easing.type:    Easing.InOutQuad
                    }
                }

                Behavior on latitude {
                    NumberAnimation {
                        duration:       editorMap.animationDuration
                        easing.type:    Easing.InOutQuad
                    }
                }

                Behavior on longitude {
                    NumberAnimation {
                        duration:       editorMap.animationDuration
                        easing.type:    Easing.InOutQuad
                    }
                }

Don Gagne's avatar
Don Gagne committed
156 157 158 159 160 161 162
                MouseArea {
                    anchors.fill: parent

                    onClicked: {
                        var coordinate = editorMap.toCoordinate(Qt.point(mouse.x, mouse.y))
                        coordinate.latitude = coordinate.latitude.toFixed(_decimalPlaces)
                        coordinate.longitude = coordinate.longitude.toFixed(_decimalPlaces)
163
                        coordinate.altitude = coordinate.altitude.toFixed(_decimalPlaces)
164 165
                        if (false /*homePositionManagerButton.checked*/) {
                            //offlineHomePosition = coordinate
166
                        } else if (addMissionItemsButton.checked) {
167
                            var index = controller.addMissionItem(coordinate)
168
                            addMissionItemsButtonAutoOffTimer.start()
169
                            setCurrentItem(index)
170 171
                        } else {
                            editorMap.zoomLevel = editorMap.maxZoomLevel - 2
172
                        }
Don Gagne's avatar
Don Gagne committed
173
                    }
Don Gagne's avatar
Don Gagne committed
174
                }
Don Gagne's avatar
Don Gagne committed
175

176
                // We use this item to support dragging since dragging a MapQuickItem just doesn't seem to work
Don Gagne's avatar
Don Gagne committed
177 178 179 180 181 182 183 184 185 186 187 188
                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
                    width:          _radius * 2
                    height:         _radius * 2
                    radius:         _radius
                    border.width:   2
                    border.color:   "white"
                    color:          "transparent"
                    visible:        false
                    z:              QGroundControl.zOrderMapItems + 1    // Above item icons
189 190 191 192 193

                    property var    missionItem
                    property var    missionItemIndicator
                    property real   heading: missionItem ? missionItem.heading : 0

Don Gagne's avatar
Don Gagne committed
194 195 196
                    readonly property real _radius:         ScreenTools.defaultFontPixelHeight * 4
                    readonly property real _arrowHeight:    ScreenTools.defaultFontPixelHeight

197
                    function clearItem() {
Don Gagne's avatar
Don Gagne committed
198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 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 246 247 248 249 250 251 252 253 254 255 256 257 258
                        itemDragger.visible = false
                        itemDragger.missionItem = undefined
                        itemDragger.missionItemIndicator = undefined
                    }

                    Image {
                        anchors.horizontalCenter:   parent.horizontalCenter
                        anchors.top:                parent.top
                        height:                     parent._arrowHeight
                        fillMode:                   Image.PreserveAspectFit
                        mipmap:                     true
                        smooth:                     true
                        source:                     "/qmlimages/ArrowHead.svg"
                    }

                    Image {
                        id:                     arrowUp
                        anchors.verticalCenter: parent.verticalCenter
                        anchors.right:          parent.right
                        height:                 parent._arrowHeight
                        fillMode:               Image.PreserveAspectFit
                        mipmap:                 true
                        smooth:                 true
                        source:                 "/qmlimages/ArrowHead.svg"
                        transform:              Rotation { origin.x: arrowUp.width / 2; origin.y: arrowUp.height / 2; angle: 90}
                    }

                    Image {
                        id:                         arrowDown
                        anchors.horizontalCenter:   parent.horizontalCenter
                        anchors.bottom:             parent.bottom
                        height:                     parent._arrowHeight
                        fillMode:                   Image.PreserveAspectFit
                        mipmap:                     true
                        smooth:                     true
                        source:                     "/qmlimages/ArrowHead.svg"
                        transform:                  Rotation { origin.x: arrowDown.width / 2; origin.y: arrowDown.height / 2; angle: 180}
                    }

                    Image {
                        id:                     arrowLeft
                        anchors.verticalCenter: parent.verticalCenter
                        anchors.left:           parent.left
                        height:                 parent._arrowHeight
                        fillMode:               Image.PreserveAspectFit
                        mipmap:                 true
                        smooth:                 true
                        source:                 "/qmlimages/ArrowHead.svg"
                        transform:              Rotation { origin.x: arrowLeft.width / 2; origin.y: arrowLeft.height / 2; angle: -90}
                    }

                    Rectangle {
                        width:                      _radius * 2
                        height:                     _radius * 2
                        radius:                     _radius
                        anchors.verticalCenter:     parent.verticalCenter
                        anchors.horizontalCenter:   parent.horizontalCenter
                        border.width:               1
                        border.color:               "white"

                        readonly property real _radius: ScreenTools.defaultFontPixelWidth / 4
259 260
                    }

261 262 263 264 265 266 267 268
                    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
269 270 271 272
                        drag.minimumX:  0
                        drag.minimumY:  0
                        drag.maximumX:  itemDragger.parent.width - parent.width
                        drag.maximumY:  itemDragger.parent.height - parent.height
273

274
                        property bool dragActive: drag.active
275

276 277
                        onDragActiveChanged: {
                            if (!drag.active) {
Don Gagne's avatar
Don Gagne committed
278
                                var point = Qt.point(itemDragger.x + (itemDragger.width  / 2), itemDragger.y + (itemDragger.height / 2))
Don Gagne's avatar
Don Gagne committed
279
                                var coordinate = editorMap.toCoordinate(point)
Don Gagne's avatar
Don Gagne committed
280 281 282 283
                                coordinate.altitude = itemDragger.missionItem.coordinate.altitude
                                itemDragger.missionItem.coordinate = coordinate
                                editorMap.latitude = itemDragger.missionItem.coordinate.latitude
                                editorMap.longitude = itemDragger.missionItem.coordinate.longitude
Don Gagne's avatar
Don Gagne committed
284 285 286
                            }
                        }
                    }
287
                }
288

Don Gagne's avatar
Don Gagne committed
289
                // Add the mission items to the map
Don Gagne's avatar
Don Gagne committed
290
                MapItemView {
291
                    model:          controller.missionItems
Don Gagne's avatar
Don Gagne committed
292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346
                    delegate:       delegateComponent
                }

                Component {
                    id: delegateComponent

                    MissionItemIndicator {
                        id:             itemIndicator
                        label:          object.homePosition ? "H" : object.sequenceNumber
                        isCurrentItem:  object.isCurrentItem
                        coordinate:     object.coordinate
                        visible:        object.specifiesCoordinate && (!object.homePosition || object.homePositionValid)
                        z:              QGroundControl.zOrderMapItems

                        onClicked: setCurrentItem(object.sequenceNumber)

                        Connections {
                            target: object

                            onIsCurrentItemChanged: {
                                if (object.isCurrentItem && object.specifiesCoordinate) {
                                    // Setup our drag item
                                    if (object.sequenceNumber != 0) {
                                        itemDragger.visible = true
                                        itemDragger.missionItem = Qt.binding(function() { return object })
                                        itemDragger.missionItemIndicator = Qt.binding(function() { return itemIndicator })
                                    } else {
                                        itemDragger.clearItem()
                                    }

                                    // Move to the new position
                                    editorMap.latitude = object.coordinate.latitude
                                    editorMap.longitude = object.coordinate.longitude
                                }
                            }
                        }

                        // 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)
                                }
                            }
                        }
                    }
347 348 349
                }

                // Add lines between waypoints
350 351
                MissionLineView {
                    model:          controller.waypointLines
Don Gagne's avatar
Don Gagne committed
352 353
                }

354
                // Mission Item Editor
Don Gagne's avatar
Don Gagne committed
355
                Item {
356 357 358 359 360
                    id:             missionItemEditor
                    anchors.top:    parent.top
                    anchors.bottom: parent.bottom
                    anchors.right:  parent.right
                    width:          _rightPanelWidth
361
                    visible:        _missionItems.count > 1
362
                    opacity:        _rightPanelOpacity
Gus Grubba's avatar
Gus Grubba committed
363
                    z:              QGroundControl.zOrderTopMost
364 365 366

                    ListView {
                        id:             missionItemSummaryList
367
                        anchors.fill:   parent
368 369
                        spacing:        _margin / 2
                        orientation:    ListView.Vertical
370
                        model:          controller.canEdit ? controller.missionItems : 0
371 372 373 374 375 376 377

                        property real _maxItemHeight: 0

                        delegate:
                            MissionItemEditor {
                            missionItem:    object
                            width:          parent.width
378 379
                            readOnly:       object.sequenceNumber == 0
                            visible:        !readOnly || object.homePositionValid
380 381 382 383 384

                            onClicked:  setCurrentItem(object.sequenceNumber)

                            onRemove: {
                                var newCurrentItem = object.sequenceNumber - 1
385
                                controller.removeMissionItem(object.sequenceNumber)
386 387 388
                                if (_missionItems.count > 1) {
                                    newCurrentItem = Math.min(_missionItems.count - 1, newCurrentItem)
                                    setCurrentItem(newCurrentItem)
389
                                }
390
                            }
391
                        }
392
                    } // ListView
Don Gagne's avatar
Don Gagne committed
393

394
                    QGCLabel {
395
                        anchors.fill:   parent
396
                        visible:        !controller.canEdit
397 398 399 400 401
                        wrapMode:       Text.WordWrap
                        text:           "The set of mission items you have loaded cannot be edited by QGroundControl. " +
                                        "You will only be able to save these to a file, or send them to a vehicle."
                    }
                } // Item - Mission Item editor
402

403 404
                /*
                  Home Position Manager temporarily disbled till more work is done on it
405

406 407 408 409 410 411 412 413 414 415
                // Home Position Manager
                Rectangle {
                    id:             homePositionManager
                    anchors.top:    parent.top
                    anchors.bottom: parent.bottom
                    anchors.right:  parent.right
                    width:          _rightPanelWidth
                    visible:        homePositionManagerButton.checked
                    color:          qgcPal.window
                    opacity:        _rightPanelOpacity
Gus Grubba's avatar
Gus Grubba committed
416
                    z:              QGroundControl.zOrderTopMost
417 418 419 420 421

                    Column {
                        anchors.margins:    _margin
                        anchors.fill:       parent
                        visible:            !liveHomePositionAvailable
422

423 424 425 426
                        QGCLabel {
                            font.pixelSize: ScreenTools.mediumFontPixelSize
                            text:           "Flying Field Manager"
                        }
427

428 429 430 431
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight
                        }
432

433 434 435 436 437
                        QGCLabel {
                            width:      parent.width
                            wrapMode:   Text.WordWrap
                            text:       "This is used to save locations associated with your flying field for use while creating missions with no vehicle connection."
                        }
438

439 440 441 442
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight
                        }
443

444 445 446
                        QGCLabel {
                            text:       "Select field to use:"
                        }
447

448 449 450 451 452 453 454 455 456 457 458 459 460
                        QGCComboBox {
                            id:         homePosCombo
                            width:      parent.width
                            textRole:   "text"
                            model:      _homePositionManager.homePositions

                            onCurrentIndexChanged: {
                                if (currentIndex != -1) {
                                    var homePos = _homePositionManager.homePositions.get(currentIndex)
                                    _homePositionName = homePos.name
                                    offlineHomePosition = homePos.coordinate
                                    editorMap.latitude = offlineHomePosition.latitude
                                    editorMap.longitude = offlineHomePosition.longitude
461 462
                                }
                            }
463
                        }
464

465 466 467 468
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight
                        }
469

470 471 472 473 474 475 476 477
                        QGCLabel {
                            width:      parent.width
                            wrapMode:   Text.WordWrap
                            text:       "To add a new flying field, click on the Map to set the position. " +
                                        "Then give it a new name and click Add/Update. " +
                                        "To change the current field position, click on the Map to set the new position. " +
                                        "Then click Add/Update without changing the name."
                        }
478

479 480 481 482
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
483

484 485 486
                        Item {
                            width:  parent.width
                            height: nameField.height
487

488 489 490
                            QGCLabel {
                                anchors.baseline:   nameField.baseline
                                text:               "Name:"
491 492
                            }

493 494 495 496 497
                            QGCTextField {
                                id:             nameField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           _homePositionName
498
                            }
499
                        }
500

501 502 503 504
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
505

506 507 508
                        Item {
                            width:  parent.width
                            height: offlineLatitudeField.height
509

510 511 512
                            QGCLabel {
                                anchors.baseline:   offlineLatitudeField.baseline
                                text:               "Lat:"
513 514
                            }

515 516 517 518 519
                            QGCTextField {
                                id:             offlineLatitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           offlineHomePosition.latitude
520
                            }
521
                        }
522

523 524 525 526
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
527

528 529 530
                        Item {
                            width:  parent.width
                            height: offlineLongitudeField.height
531

532 533 534
                            QGCLabel {
                                anchors.baseline:   offlineLongitudeField.baseline
                                text:               "Lon:"
535 536
                            }

537 538 539 540 541
                            QGCTextField {
                                id:             offlineLongitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           offlineHomePosition.longitude
542
                            }
543
                        }
544

545 546 547 548
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
549

550 551 552
                        Item {
                            width:  parent.width
                            height: offlineAltitudeField.height
553

554 555 556
                            QGCLabel {
                                anchors.baseline:   offlineAltitudeField.baseline
                                text:               "Alt:"
557 558
                            }

559 560 561 562 563
                            QGCTextField {
                                id:             offlineAltitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           offlineHomePosition.altitude
564
                            }
565
                        }
566

567 568 569 570
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight
                        }
571

572 573
                        Row {
                            spacing: ScreenTools.defaultFontPixelWidth
574

575 576
                            QGCButton {
                                text: "Add/Update"
577

578 579 580 581
                                onClicked: {
                                    offlineHomePosition = QtPositioning.coordinate(latitudeField.text, longitudeField.text, altitudeField.text)
                                    _homePositionManager.updateHomePosition(nameField.text, offlineHomePosition)
                                    homePosCombo.currentIndex = homePosCombo.find(nameField.text)
582 583
                                }
                            }
584

585 586
                            QGCButton {
                                text: "Delete"
587

588 589 590 591 592 593 594 595
                                onClicked: {
                                    homePosCombo.currentIndex = -1
                                    _homePositionManager.deleteHomePosition(nameField.text)
                                    homePosCombo.currentIndex = 0
                                    var homePos = _homePositionManager.homePositions.get(0)
                                    _homePositionName = homePos.name
                                    offlineHomePosition = homePos.coordinate
                                }
596
                            }
597 598
                        }
                    } // Column - Offline view
599

600 601 602 603
                    Column {
                        anchors.margins:    _margin
                        anchors.fill:       parent
                        visible:            liveHomePositionAvailable
604

605 606 607 608
                        QGCLabel {
                            font.pixelSize: ScreenTools.mediumFontPixelSize
                            text:           "Vehicle Home Position"
                        }
609

610 611 612 613
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight
                        }
614

615 616 617 618 619 620 621
                        Item {
                            width:  parent.width
                            height: liveLatitudeField.height

                            QGCLabel {
                                anchors.baseline:   liveLatitudeField.baseline
                                text:               "Lat:"
622 623
                            }

624 625 626 627 628
                            QGCLabel {
                                id:             liveLatitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           liveHomePosition.latitude
629
                            }
630
                        }
631

632 633 634 635
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
636

637 638 639
                        Item {
                            width:  parent.width
                            height: liveLongitudeField.height
640

641 642 643
                            QGCLabel {
                                anchors.baseline:   liveLongitudeField.baseline
                                text:               "Lon:"
644 645
                            }

646 647 648 649 650
                            QGCLabel {
                                id:             liveLongitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           liveHomePosition.longitude
651
                            }
652
                        }
653

654 655 656 657
                        Item {
                            width: 10
                            height: ScreenTools.defaultFontPixelHeight / 3
                        }
658

659 660 661
                        Item {
                            width:  parent.width
                            height: liveAltitudeField.height
662

663 664 665
                            QGCLabel {
                                anchors.baseline:   liveAltitudeField.baseline
                                text:               "Alt:"
666 667
                            }

668 669 670 671 672 673 674 675 676
                            QGCLabel {
                                id:             liveAltitudeField
                                anchors.right:  parent.right
                                width:          _editFieldWidth
                                text:           liveHomePosition.altitude
                            }
                        }
                    } // Column - Online view
                } // Item - Home Position Manager
677
                */
678 679 680

                // Help Panel
                Rectangle {
Don Gagne's avatar
Don Gagne committed
681 682 683 684
                    id:                 helpPanel
                    anchors.margins:    margins
                    anchors.top:        parent.top
                    anchors.bottom:     parent.bottom
685 686
                    anchors.left:       addMissionItemsButton.right
                    anchors.right:      missionItemEditor.left
Don Gagne's avatar
Don Gagne committed
687 688 689 690 691
                    width:              parent.width - (margins * 2) - _rightPanelWidth
                    visible:            helpButton.checked
                    color:              qgcPal.window
                    opacity:            _rightPanelOpacity
                    radius:             ScreenTools.defaultFontPixelHeight
Gus Grubba's avatar
Gus Grubba committed
692
                    z:                  QGroundControl.zOrderTopMost
Don Gagne's avatar
Don Gagne committed
693

694
                    readonly property real margins:  ScreenTools.defaultFontPixelHeight
Don Gagne's avatar
Don Gagne committed
695 696 697 698 699 700 701

                    Image {
                        anchors.margins:    ScreenTools.defaultFontPixelHeight
                        anchors.top:        parent.top
                        anchors.right:      parent.right
                        width:              ScreenTools.defaultFontPixelHeight * 1.5
                        height:             ScreenTools.defaultFontPixelHeight * 1.5
702
                        source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/XDeleteBlack.svg" : "/qmlimages/XDelete.svg"
Don Gagne's avatar
Don Gagne committed
703 704 705 706 707 708 709 710 711
                        fillMode:           Image.PreserveAspectFit
                        mipmap:             true
                        smooth:             true

                        MouseArea {
                            anchors.fill:   parent
                            onClicked:      helpButton.checked = false
                        }
                    }
712

713
                    Item {
714 715
                        anchors.margins:    _margin
                        anchors.fill:       parent
716 717 718 719 720 721 722 723 724 725 726 727 728

                        QGCLabel {
                            id:             helpTitle
                            font.pixelSize: ScreenTools.mediumFontPixelSize
                            text:           "Mission Planner"
                        }

                        QGCLabel {
                            id:                 helpIconLabel
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        helpTitle.bottom
                            width:              parent.width
                            wrapMode:           Text.WordWrap
dogmaphobic's avatar
dogmaphobic committed
729
                            text:               "Mission Planner tool buttons:"
730 731
                        }

dogmaphobic's avatar
dogmaphobic committed
732 733 734 735 736 737 738 739
                        Image {
                            id:                 addMissionItemsHelpIcon
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        helpIconLabel.bottom
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
740
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/MapAddMissionBlack.svg" : "/qmlimages/MapAddMission.svg"
741 742 743 744 745 746 747 748 749
                        }

                        QGCLabel {
                            id:                 addMissionItemsHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        addMissionItemsHelpIcon.top
                            wrapMode:           Text.WordWrap
dogmaphobic's avatar
dogmaphobic committed
750 751
                            text:               "<b>Add Mission Items</b><br>" +
                                                "When enabled, add mission items by clicking on the map."
752 753
                        }

dogmaphobic's avatar
dogmaphobic committed
754
                        Image {
Don Gagne's avatar
Don Gagne committed
755
                            id:                 deleteHelpIcon
dogmaphobic's avatar
dogmaphobic committed
756 757 758 759 760 761
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        addMissionItemsHelpText.bottom
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
762
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/TrashDeleteBlack.svg" : "/qmlimages/TrashDelete.svg"
Don Gagne's avatar
Don Gagne committed
763 764 765 766 767 768 769 770 771 772 773 774 775
                        }

                        QGCLabel {
                            id:                 deleteHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        deleteHelpIcon.top
                            wrapMode:           Text.WordWrap
                            text:               "<b>Delete Mission Item</b><br>" +
                                                "Delete the currently selected mission item."
                        }

776
                        /*
777
                          Home Position Manager disabled
778

Don Gagne's avatar
Don Gagne committed
779 780 781 782 783 784 785 786
                        Image {
                            id:                 homePositionManagerHelpIcon
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        deleteHelpText.bottom
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
787
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/MapHomeBlack.svg" : "/qmlimages/MapHome.svg"
788 789 790 791 792 793 794 795 796
                        }

                        QGCLabel {
                            id:                 homePositionManagerHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        homePositionManagerHelpIcon.top
                            wrapMode:           Text.WordWrap
797 798 799
                            text:               "<b>Flying Field Manager</b><br>" +
                                                "When enabled, allows you to select/add/update flying field locations. " +
                                                "You can save multiple flying field locations for use while creating missions while you are not connected to your vehicle."
800
                        }
801
                        */
802 803 804 805

                        Image {
                            id:                 mapCenterHelpIcon
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
806
                            anchors.top:        deleteHelpText.bottom
807 808 809 810
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
811
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/MapCenterBlack.svg" : "/qmlimages/MapCenter.svg"
812 813 814 815 816 817 818 819 820
                        }

                        QGCLabel {
                            id:                 mapCenterHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        mapCenterHelpIcon.top
                            wrapMode:           Text.WordWrap
dogmaphobic's avatar
dogmaphobic committed
821 822
                            text:               "<b>Map Center</b><br>" +
                                                "Options for centering the map."
823 824 825 826 827 828 829 830 831 832
                        }

                        Image {
                            id:                 syncHelpIcon
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        mapCenterHelpText.bottom
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
833
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/MapSyncBlack.svg" : "/qmlimages/MapSync.svg"
834 835 836 837 838 839 840 841 842
                        }

                        QGCLabel {
                            id:                 syncHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        syncHelpIcon.top
                            wrapMode:           Text.WordWrap
dogmaphobic's avatar
dogmaphobic committed
843 844
                            text:               "<b>Sync</b><br>" +
                                                "Options for saving/loading mission items."
845 846 847 848 849 850 851 852 853 854
                        }

                        Image {
                            id:                 mapTypeHelpIcon
                            anchors.topMargin:  ScreenTools.defaultFontPixelHeight
                            anchors.top:        syncHelpText.bottom
                            width:              ScreenTools.defaultFontPixelHeight * 3
                            fillMode:           Image.PreserveAspectFit
                            mipmap:             true
                            smooth:             true
855
                            source:             (qgcPal.globalTheme === QGCPalette.Light) ? "/qmlimages/MapTypeBlack.svg" : "/qmlimages/MapType.svg"
856 857 858 859 860 861 862 863 864
                        }

                        QGCLabel {
                            id:                 mapTypeHelpText
                            anchors.leftMargin: ScreenTools.defaultFontPixelHeight
                            anchors.left:       mapTypeHelpIcon.right
                            anchors.right:      parent.right
                            anchors.top:        mapTypeHelpIcon.top
                            wrapMode:           Text.WordWrap
dogmaphobic's avatar
dogmaphobic committed
865 866
                            text:               "<b>Map Type</b><br>" +
                                                "Map type options."
867
                        }
Don Gagne's avatar
Don Gagne committed
868 869

                        QGCCheckBox {
870 871 872 873 874
                            anchors.left:       parent.left
                            anchors.bottom:     parent.bottom
                            anchors.margins:    _margin
                            checked:            !_showHelp
                            text:               "Don't show me again"
Don Gagne's avatar
Don Gagne committed
875

876
                            onClicked:          QGroundControl.flightMapSettings.saveBoolMapSetting(editorMap.mapName, _showHelpKey, !checked)
Don Gagne's avatar
Don Gagne committed
877
                        }
878 879 880 881
                    } // Item - margin
                } // Item - Help Panel

                RoundButton {
882 883 884 885 886 887
                    id:                 addMissionItemsButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    y:                  (parent.height - (_toolButtonCount * height) - ((_toolButtonCount - 1) * _margin)) / 2
                    buttonImage:        "/qmlimages/MapAddMission.svg"
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
888
                    z:                  QGroundControl.zOrderWidgets
889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907

                    onCheckedChanged: {
                        if (checked) {
                            addMissionItemsButtonAutoOffTimer.start()
                        } else {
                            addMissionItemsButtonAutoOffTimer.stop()
                        }
                    }

                    Timer {
                        id:         addMissionItemsButtonAutoOffTimer
                        interval:   _addMissionItemsButtonAutoOffTimeout
                        repeat:     false

                        onTriggered: addMissionItemsButton.checked = false
                    }
                }

                RoundButton {
908
                    id:                 deleteMissionItemButton
909 910 911
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    anchors.top:        addMissionItemsButton.bottom
912 913
                    buttonImage:        "/qmlimages/TrashDelete.svg"
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
914
                    z:                  QGroundControl.zOrderWidgets
915 916

                    onClicked: {
Don Gagne's avatar
Don Gagne committed
917
                        itemDragger.clearItem()
918 919 920 921 922
                        controller.deleteCurrentMissionItem()
                        checked = false
                    }
                }

923
                /*
924
                  Home Position manager temporarily disable
925 926 927 928 929
                RoundButton {
                    id:                 homePositionManagerButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    anchors.top:        deleteMissionItemButton.bottom
930 931
                    buttonImage:        "/qmlimages/MapHome.svg"
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
932
                    z:                  QGroundControl.zOrderWidgets
933
                }
934
                */
935 936 937 938 939

                DropButton {
                    id:                 centerMapButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
940
                    anchors.top:        deleteMissionItemButton.bottom
941 942 943 944
                    dropDirection:      dropRight
                    buttonImage:        "/qmlimages/MapCenter.svg"
                    viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
945
                    z:                  QGroundControl.zOrderWidgets
946 947 948 949 950 951 952 953 954

                    dropDownComponent: Component {
                        Column {
                            QGCLabel { text: "Center map:" }

                            Row {
                                spacing: ScreenTools.defaultFontPixelWidth

                                QGCButton {
955 956
                                    text:       "Home"
                                    enabled:    liveHomePositionAvailable
957 958 959

                                    onClicked: {
                                        centerMapButton.hideDropDown()
960
                                        editorMap.center = liveHomePosition
961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989
                                    }
                                }

                                QGCButton {
                                    text:       "Vehicle"
                                    enabled:    activeVehicle && activeVehicle.latitude != 0 && activeVehicle.longitude != 0

                                    property var activeVehicle: multiVehicleManager.activeVehicle

                                    onClicked: {
                                        centerMapButton.hideDropDown()
                                        editorMap.latitude = activeVehicle.latitude
                                        editorMap.longitude = activeVehicle.longitude
                                    }
                                }
                            }
                        }
                    }
                }

                DropButton {
                    id:                 syncButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    anchors.top:        centerMapButton.bottom
                    dropDirection:      dropRight
                    buttonImage:        _syncNeeded ? "/qmlimages/MapSyncChanged.svg" : "/qmlimages/MapSync.svg"
                    viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
990
                    z:                  QGroundControl.zOrderWidgets
991 992
                    dropDownComponent:  syncDropDownComponent
                    enabled:            !_syncInProgress
993 994 995 996 997 998 999 1000 1001 1002 1003
                }

                DropButton {
                    id:                 mapTypeButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    anchors.top:        syncButton.bottom
                    dropDirection:      dropRight
                    buttonImage:        "/qmlimages/MapType.svg"
                    viewportMargins:    ScreenTools.defaultFontPixelWidth / 2
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
1004
                    z:                  QGroundControl.zOrderWidgets
1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040

                    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()
                                        }
                                    }
                                }
                            }
                        }
                    }
                }

                RoundButton {
                    id:                 helpButton
                    anchors.margins:    _margin
                    anchors.left:       parent.left
                    anchors.top:        mapTypeButton.bottom
                    buttonImage:        "/qmlimages/Help.svg"
                    exclusiveGroup:     _dropButtonsExclusiveGroup
Gus Grubba's avatar
Gus Grubba committed
1041
                    z:                  QGroundControl.zOrderWidgets
Don Gagne's avatar
Don Gagne committed
1042
                    checked:            _showHelp
1043 1044
                }
            } // FlightMap
Don Gagne's avatar
Don Gagne committed
1045 1046
        } // Item - split view container
    } // QGCViewPanel
1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063

    Component {
        id: syncDropDownComponent

        Column {
            id:         columnHolder
            spacing:    _margin

            QGCLabel {
                width:      columnHolder.width
                wrapMode:   Text.WordWrap
                text:       _syncNeeded && !controller.autoSync ?
                                "You have unsaved changed to you mission. You should send to your vehicle, or save to a file:" :
                                "Sync:"
            }

            Row {
1064
                visible:    true //autoSyncCheckBox.enabled && autoSyncCheckBox.checked
1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108
                spacing:    ScreenTools.defaultFontPixelWidth

                QGCButton {
                    text:       "Send to vehicle"
                    enabled:    _activeVehicle && !_activeVehicle.missionManager.inProgress

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

                QGCButton {
                    text:       "Load from vehicle"
                    enabled:    _activeVehicle && !_activeVehicle.missionManager.inProgress

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

            Row {
                spacing: ScreenTools.defaultFontPixelWidth

                QGCButton {
                    text:       "Save to file..."

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

                QGCButton {
                    text:       "Load from file..."

                    onClicked: {
                        syncButton.hideDropDown()
                        controller.loadMissionFromFile()
                    }
                }
            }
1109 1110
/*
        FIXME: autoSync is temporarily disconnected since it's still buggy
1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125

            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
            }
1126
*/
1127 1128
        }
    }
Don Gagne's avatar
Don Gagne committed
1129
} // QGCVIew