MainWindow.cc 48 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009 - 2011 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/>.

pixhawk's avatar
pixhawk committed
22 23 24 25
======================================================================*/

/**
 * @file
26
 *   @brief Implementation of class MainWindow
27
 *   @author Lorenz Meier <mail@qgroundcontrol.org>
pixhawk's avatar
pixhawk committed
28 29 30 31 32 33 34 35 36 37
 */

#include <QSettings>
#include <QDockWidget>
#include <QNetworkInterface>
#include <QMessageBox>
#include <QDebug>
#include <QTimer>
#include <QHostInfo>

38
#include "QGC.h"
pixhawk's avatar
pixhawk committed
39 40 41 42 43
#include "MAVLinkSimulationLink.h"
#include "SerialLink.h"
#include "UDPLink.h"
#include "MAVLinkProtocol.h"
#include "CommConfigurationWindow.h"
44
#include "QGCWaypointListMulti.h"
pixhawk's avatar
pixhawk committed
45 46
#include "MainWindow.h"
#include "JoystickWidget.h"
pixhawk's avatar
pixhawk committed
47
#include "GAudioOutput.h"
48
#include "QGCToolWidget.h"
49
#include "QGCMAVLinkLogPlayer.h"
50
#include "QGCSettingsWidget.h"
51
#include "QGCMapTool.h"
52

53
#ifdef QGC_OSG_ENABLED
54
#include "Q3DWidgetFactory.h"
55
#endif
pixhawk's avatar
pixhawk committed
56

lm's avatar
lm committed
57 58 59 60
// FIXME Move
#include "PxQuadMAV.h"
#include "SlugsMAV.h"

pixhawk's avatar
pixhawk committed
61

62
#include "LogCompressor.h"
pixhawk's avatar
pixhawk committed
63

64 65
MainWindow* MainWindow::instance()
{
66
    static MainWindow* _instance = 0;
67 68
    if(_instance == 0)
    {
69
        _instance = new MainWindow();
70

71
        /* Set the application as parent to ensure that this object
72
                 * will be destroyed when the main application exits */
73 74 75
        //_instance->setParent(qApp);
    }
    return _instance;
76 77
}

pixhawk's avatar
pixhawk committed
78 79 80 81 82 83 84
/**
* Create new mainwindow. The constructor instantiates all parts of the user
* interface. It does NOT show the mainwindow. To display it, call the show()
* method.
*
* @see QMainWindow::show()
**/
85
MainWindow::MainWindow(QWidget *parent):
86 87 88 89 90 91 92
    QMainWindow(parent),
    toolsMenuActions(),
    currentView(VIEW_UNCONNECTED),
    aboutToCloseFlag(false),
    changingViewsFlag(false),
    styleFileName(QCoreApplication::applicationDirPath() + "/style-indoor.css"),
    autoReconnect(false),
93
    currentStyle(QGC_MAINWINDOW_STYLE_INDOOR),
94 95
    lowPowerMode(false),
    centerStackActionGroup(this)
pixhawk's avatar
pixhawk committed
96
{
97
    hide();
98
    loadSettings();
99 100
    if (!settings.contains("CURRENT_VIEW"))
    {
101 102
        // Set this view as default view
        settings.setValue("CURRENT_VIEW", currentView);
103 104 105
    }
    else
    {
106 107 108
        // LOAD THE LAST VIEW
        VIEW_SECTIONS currentViewCandidate = (VIEW_SECTIONS) settings.value("CURRENT_VIEW", currentView).toInt();
        if (currentViewCandidate != VIEW_ENGINEER &&
109
                currentViewCandidate != VIEW_OPERATOR &&
110 111 112
                currentViewCandidate != VIEW_PILOT &&
                currentViewCandidate != VIEW_FULL)
        {
113
            currentView = currentViewCandidate;
114
        }
115 116
    }

117
    setDefaultSettingsForAp();
118

119 120
    settings.sync();

121 122 123
    // Setup UI state machines
    centerStackActionGroup.setExclusive(true);

pixhawk's avatar
pixhawk committed
124 125 126
    // Setup user interface
    ui.setupUi(this);

127 128
    centerStack = new QStackedWidget(this);
    setCentralWidget(centerStack);
129

lm's avatar
lm committed
130 131 132 133 134 135 136
    // Load Toolbar
    toolBar = new QGCToolBar(this);
    this->addToolBar(toolBar);
    // Add actions
    toolBar->addPerspectiveChangeAction(ui.actionOperatorsView);
    toolBar->addPerspectiveChangeAction(ui.actionEngineersView);
    toolBar->addPerspectiveChangeAction(ui.actionPilotsView);
137
//    toolBar->addPerspectiveChangeAction(ui.actionUnconnectedView);
lm's avatar
lm committed
138

139 140
    buildCommonWidgets();
    connectCommonWidgets();
141 142

    configureWindowName();
pixhawk's avatar
pixhawk committed
143

144
    loadStyle(currentStyle);
145

146
    // Create actions
147
    connectCommonActions();
148

149 150 151
    // Set dock options
    setDockOptions(AnimatedDocks | AllowTabbedDocks | AllowNestedDocks);

152
    // Load mavlink view as default widget set
153
    //loadMAVLinkView();
154

lm's avatar
lm committed
155 156
    statusBar()->setSizeGripEnabled(true);

157 158 159 160 161 162
    // Restore the window setup
    if (settings.contains(getWindowStateKey()))
    {
        loadViewState();
    }

163
    // Restore the window position and size
164 165
    if (settings.contains(getWindowGeometryKey()))
    {
166
        // Restore the window geometry
167
        restoreGeometry(settings.value(getWindowGeometryKey()).toByteArray());
168 169 170
    }
    else
    {
171 172 173
        // Adjust the size
        adjustSize();
    }
pixhawk's avatar
pixhawk committed
174

175 176
    // Populate link menu
    QList<LinkInterface*> links = LinkManager::instance()->getLinks();
177 178
    foreach(LinkInterface* link, links)
    {
179 180
        this->addLink(link);
    }
181

182 183
    connect(LinkManager::instance(), SIGNAL(newLink(LinkInterface*)), this, SLOT(addLink(LinkInterface*)));

184
    // Connect user interface devices
185 186
    joystickWidget = 0;
    joystick = new JoystickInput();
187

lm's avatar
lm committed
188 189 190 191 192
    // Connect flighgear test link
    // FIXME MOVE INTO UAV OBJECT
    fgLink = new QGCFlightGearLink();
    fgLink->connectSimulation();

193
    // Connect link
194 195
    if (autoReconnect)
    {
196 197 198 199 200 201
        SerialLink* link = new SerialLink();
        // Add to registry
        LinkManager::instance()->add(link);
        LinkManager::instance()->addProtocol(link, mavlink);
        link->connect();
    }
202

203 204 205
    // Set low power mode
    enableLowPowerMode(lowPowerMode);

206 207
    // Initialize window state
    windowStateVal = windowState();
208 209

    show();
pixhawk's avatar
pixhawk committed
210 211
}

pixhawk's avatar
pixhawk committed
212
MainWindow::~MainWindow()
pixhawk's avatar
pixhawk committed
213
{
214
    delete mavlink;
215
    delete joystick;
lm's avatar
lm committed
216

217 218 219 220 221 222
    // Get and delete all dockwidgets and contained
    // widgets
    QObjectList childList( this->children() );

    QObjectList::iterator i;
    QDockWidget* dockWidget;
223 224
    for (i = childList.begin(); i != childList.end(); ++i)
    {
225
        dockWidget = dynamic_cast<QDockWidget*>(*i);
226 227
        if (dockWidget)
        {
228 229 230 231 232
            // Remove dock widget from main window
            removeDockWidget(dockWidget);
            delete dockWidget->widget();
            delete dockWidget;
        }
LM's avatar
LM committed
233 234 235 236
        else
        {
            delete dynamic_cast<QObject*>(*i);
        }
237
    }
LM's avatar
LM committed
238 239

    // Delete all UAS objects
pixhawk's avatar
pixhawk committed
240 241
}

242 243 244 245 246
/**
 * Set default settings for this AP type.
 */
void MainWindow::setDefaultSettingsForAp()
{
LM's avatar
LM committed
247
    // Check if the settings exist, instantiate defaults if necessary
248

249

250 251 252 253
//    // UNCONNECTED VIEW DEFAULT
////    QString centralKey = buildMenuKey(SUB_SECTION_CHECKED, CENTRAL_MAP, VIEW_UNCONNECTED);
//    if (!settings.contains(centralKey)) {
//        settings.setValue(centralKey,true);
254

255 256 257 258 259
//        // ENABLE UAS LIST
////        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_UAS_LIST, VIEW_UNCONNECTED), true);
//        // ENABLE COMMUNICATION CONSOLE
////        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_DEBUG_CONSOLE, VIEW_UNCONNECTED), true);
//    }
260

261 262 263 264 265 266 267 268 269 270 271 272
//    // OPERATOR VIEW DEFAULT
////    centralKey = buildMenuKey(SUB_SECTION_CHECKED, CENTRAL_MAP, VIEW_OPERATOR);
//    if (!settings.contains(centralKey)) {
////        settings.setValue(centralKey,true);

//        // ENABLE UAS LIST
////        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_UAS_LIST,VIEW_OPERATOR), true);
//        // ENABLE HUD TOOL WIDGET
////        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_HUD,VIEW_OPERATOR), true);
//        // ENABLE WAYPOINTS
////        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_WAYPOINTS,VIEW_OPERATOR), true);
//    }
273

274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
//    // ENGINEER VIEW DEFAULT
//    centralKey = buildMenuKey(SUB_SECTION_CHECKED, CENTRAL_LINECHART, VIEW_ENGINEER);
//    if (!settings.contains(centralKey)) {
//        settings.setValue(centralKey,true);
//        // Enable Parameter widget
//        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_PARAMETERS,VIEW_ENGINEER), true);
//    }

//    // MAVLINK VIEW DEFAULT
//    centralKey = buildMenuKey(SUB_SECTION_CHECKED, CENTRAL_PROTOCOL, VIEW_MAVLINK);
//    if (!settings.contains(centralKey)) {
//        settings.setValue(centralKey,true);
//    }

//    // PILOT VIEW DEFAULT
//    centralKey = buildMenuKey(SUB_SECTION_CHECKED, CENTRAL_HUD, VIEW_PILOT);
//    if (!settings.contains(centralKey)) {
//        settings.setValue(centralKey,true);
//        // Enable Flight display
//        settings.setValue(buildMenuKey(SUB_SECTION_CHECKED,MainWindow::MENU_HDD_1,VIEW_PILOT), true);
//    }
295 296
}

lm's avatar
lm committed
297 298
void MainWindow::resizeEvent(QResizeEvent * event)
{
299 300
    if (height() < 800)
    {
lm's avatar
lm committed
301
        ui.statusBar->setVisible(false);
302 303 304
    }
    else
    {
lm's avatar
lm committed
305
        ui.statusBar->setVisible(true);
306
        ui.statusBar->setSizeGripEnabled(true);
lm's avatar
lm committed
307
    }
308 309 310 311 312 313 314 315 316 317 318

    if (width() > 1200)
    {
        toolBar->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
    }
    else
    {
        toolBar->setToolButtonStyle(Qt::ToolButtonIconOnly);
    }

    QMainWindow::resizeEvent(event);
lm's avatar
lm committed
319 320
}

321 322
QString MainWindow::getWindowStateKey()
{
323
    return QString::number(currentView)+"_windowstate";
324 325 326 327
}

QString MainWindow::getWindowGeometryKey()
{
328 329
    //return QString::number(currentView)+"_geometry";
    return "_geometry";
330 331
}

lm's avatar
lm committed
332 333 334
void MainWindow::buildCustomWidget()
{
    // Show custom widgets only if UAS is connected
335 336
    if (UASManager::instance()->getActiveUAS() != NULL)
    {
lm's avatar
lm committed
337 338 339
        // Create custom widgets
        QList<QGCToolWidget*> widgets = QGCToolWidget::createWidgetsFromSettings(this);

340 341
        if (widgets.size() > 0)
        {
lm's avatar
lm committed
342 343 344
            ui.menuTools->addSeparator();
        }

345 346
        for(int i = 0; i < widgets.size(); ++i)
        {
lm's avatar
lm committed
347
            // Check if this widget already has a parent, do not create it in this case
348 349
            QGCToolWidget* tool = widgets.at(i);
            QDockWidget* dock = dynamic_cast<QDockWidget*>(tool->parentWidget());
350 351
            if (!dock)
            {
352 353 354 355
                QDockWidget* dock = new QDockWidget(tool->windowTitle(), this);
                dock->setObjectName(tool->objectName()+"_DOCK");
                dock->setWidget(tool);
                connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
lm's avatar
lm committed
356
                QAction* showAction = new QAction(widgets.at(i)->windowTitle(), this);
357
                showAction->setCheckable(true);
lm's avatar
lm committed
358 359 360 361
                connect(showAction, SIGNAL(triggered(bool)), dock, SLOT(setVisible(bool)));
                connect(dock, SIGNAL(visibilityChanged(bool)), showAction, SLOT(setChecked(bool)));
                widgets.at(i)->setMainMenuAction(showAction);
                ui.menuTools->addAction(showAction);
362 363 364 365 366

                // Load dock widget location (default is bottom)
                Qt::DockWidgetArea location = static_cast <Qt::DockWidgetArea>(tool->getDockWidgetArea(currentView));

                addDockWidget(location, dock);
lm's avatar
lm committed
367 368 369 370 371
            }
        }
    }
}

372 373 374 375
void MainWindow::buildCommonWidgets()
{
    //TODO:  move protocol outside UI
    mavlink     = new MAVLinkProtocol();
376
    connect(mavlink, SIGNAL(protocolStatusMessage(QString,QString)), this, SLOT(showCriticalMessage(QString,QString)), Qt::QueuedConnection);
377 378

    // Dock widgets
379 380
    if (!controlDockWidget)
    {
381
        controlDockWidget = new QDockWidget(tr("Control"), this);
382
        controlDockWidget->setObjectName("UNMANNED_SYSTEM_CONTROL_DOCKWIDGET");
383
        controlDockWidget->setWidget( new UASControlWidget(this) );
384
        addTool(controlDockWidget, tr("Control"), Qt::LeftDockWidgetArea);
385
    }
386

387 388
    if (!listDockWidget)
    {
389 390
        listDockWidget = new QDockWidget(tr("Unmanned Systems"), this);
        listDockWidget->setWidget( new UASListWidget(this) );
391
        listDockWidget->setObjectName("UNMANNED_SYSTEMS_LIST_DOCKWIDGET");
392
        addTool(listDockWidget, tr("Unmanned Systems"), Qt::RightDockWidgetArea);
393
    }
394

395 396
    if (!waypointsDockWidget)
    {
397
        waypointsDockWidget = new QDockWidget(tr("Mission Plan"), this);
398
        waypointsDockWidget->setWidget( new QGCWaypointListMulti(this) );
399
        waypointsDockWidget->setObjectName("WAYPOINT_LIST_DOCKWIDGET");
400
        addTool(waypointsDockWidget, tr("Mission Plan"), Qt::BottomDockWidgetArea);
401
    }
402

403 404
    if (!infoDockWidget)
    {
405 406
        infoDockWidget = new QDockWidget(tr("Status Details"), this);
        infoDockWidget->setWidget( new UASInfoWidget(this) );
pixhawk's avatar
pixhawk committed
407
        infoDockWidget->setObjectName("UAS_STATUS_DETAILS_DOCKWIDGET");
408
        addTool(infoDockWidget, tr("Status Details"), Qt::RightDockWidgetArea);
409
    }
410

411 412
    if (!debugConsoleDockWidget)
    {
413 414
        debugConsoleDockWidget = new QDockWidget(tr("Communication Console"), this);
        debugConsoleDockWidget->setWidget( new DebugConsole(this) );
415
        debugConsoleDockWidget->setObjectName("COMMUNICATION_DEBUG_CONSOLE_DOCKWIDGET");
416
        addTool(debugConsoleDockWidget, tr("Communication Console"), Qt::BottomDockWidgetArea);
417
    }
418

419 420
    if (!logPlayerDockWidget)
    {
421
        logPlayerDockWidget = new QDockWidget(tr("MAVLink Log Player"), this);
lm's avatar
lm committed
422 423 424
        logPlayer = new QGCMAVLinkLogPlayer(mavlink, this);
        toolBar->setLogPlayer(logPlayer);
        logPlayerDockWidget->setWidget(logPlayer);
425
        logPlayerDockWidget->setObjectName("MAVLINK_LOG_PLAYER_DOCKWIDGET");
426
        addTool(logPlayerDockWidget, tr("MAVLink Log Replay"), Qt::RightDockWidgetArea);
427 428
    }

429
    // Center widgets
430 431
    if (!mapWidget)
    {
432
        mapWidget = new QGCMapTool(this);
433
        addCentralWidget(mapWidget, "Maps");
434
    }
435

436 437
    if (!protocolWidget)
    {
438
        protocolWidget    = new XMLCommProtocolWidget(this);
439
        addCentralWidget(protocolWidget, "Mavlink Generator");
440
    }
lm's avatar
lm committed
441

442 443
    if (!dataplotWidget)
    {
lm's avatar
lm committed
444
        dataplotWidget    = new QGCDataPlot2D(this);
445
        addCentralWidget(dataplotWidget, "Logfile Plot");
lm's avatar
lm committed
446
    }
447

pixhawk's avatar
pixhawk committed
448 449
    //FIXME: memory of acceptList will never be freed again
    QStringList* acceptList = new QStringList();
pixhawk's avatar
pixhawk committed
450 451 452
    acceptList->append("-105,roll deg,deg,+105,s");
    acceptList->append("-105,pitch deg,deg,+105,s");
    acceptList->append("-105,heading deg,deg,+105,s");
453

pixhawk's avatar
pixhawk committed
454 455 456
    acceptList->append("-60,rollspeed d/s,deg/s,+60,s");
    acceptList->append("-60,pitchspeed d/s,deg/s,+60,s");
    acceptList->append("-60,yawspeed d/s,deg/s,+60,s");
457
    acceptList->append("0,airspeed,m/s,30");
458 459
    acceptList->append("0,groundspeed,m/s,30");
    acceptList->append("0,climbrate,m/s,30");
460
    acceptList->append("0,throttle,%,100");
461

pixhawk's avatar
pixhawk committed
462 463
    //FIXME: memory of acceptList2 will never be freed again
    QStringList* acceptList2 = new QStringList();
464 465 466 467 468 469 470 471
    acceptList2->append("900,servo #1,us,2100,s");
    acceptList2->append("900,servo #2,us,2100,s");
    acceptList2->append("900,servo #3,us,2100,s");
    acceptList2->append("900,servo #4,us,2100,s");
    acceptList2->append("900,servo #5,us,2100,s");
    acceptList2->append("900,servo #6,us,2100,s");
    acceptList2->append("900,servo #7,us,2100,s");
    acceptList2->append("900,servo #8,us,2100,s");
lm's avatar
lm committed
472
    acceptList2->append("0,abs pressure,hPa,65500");
473 474 475
    //acceptList2->append("-2048,accel. x,raw,2048,s");
    //acceptList2->append("-2048,accel. y,raw,2048,s");
    //acceptList2->append("-2048,accel. z,raw,2048,s");
476

477
    if (!hudWidget) {
478
        hudWidget         = new HUD(320, 240, this);
479
        addCentralWidget(hudWidget, tr("Head Up Display"));
480
    }
481

482
    if (!dataplotWidget) {
483
        dataplotWidget    = new QGCDataPlot2D(this);
484
        addCentralWidget(dataplotWidget, tr("Logfile Plot"));
485
    }
486

487
#ifdef QGC_OSG_ENABLED
488
    if (!_3DWidget) {
489
        _3DWidget         = Q3DWidgetFactory::get("PIXHAWK");
490
        addCentralWidget(_3DWidget, tr("Local 3D"));
491
    }
492
#endif
lm's avatar
lm committed
493

494
#if (defined _MSC_VER) | (defined Q_OS_MAC)
495
    if (!gEarthWidget) {
496
        gEarthWidget = new QGCGoogleEarthView(this);
497
        addCentralWidget(gEarthWidget, tr("Google Earth"));
498
    }
499

500
#endif
501

pixhawk's avatar
pixhawk committed
502
    // Dock widgets
503

504
    if (!detectionDockWidget) {
505 506
        detectionDockWidget = new QDockWidget(tr("Object Recognition"), this);
        detectionDockWidget->setWidget( new ObjectDetectionView("images/patterns", this) );
pixhawk's avatar
pixhawk committed
507
        detectionDockWidget->setObjectName("OBJECT_DETECTION_DOCK_WIDGET");
508
        addTool(detectionDockWidget, tr("Object Recognition"), Qt::RightDockWidgetArea);
509
    }
510

511
    if (!parametersDockWidget) {
512
        parametersDockWidget = new QDockWidget(tr("Calibration and Onboard Parameters"), this);
513
        parametersDockWidget->setWidget( new ParameterInterface(this) );
pixhawk's avatar
pixhawk committed
514
        parametersDockWidget->setObjectName("PARAMETER_INTERFACE_DOCKWIDGET");
515
        addTool(parametersDockWidget, tr("Calibration and Parameters"), Qt::RightDockWidgetArea);
516
    }
517

518
    if (!watchdogControlDockWidget) {
519 520
        watchdogControlDockWidget = new QDockWidget(tr("Process Control"), this);
        watchdogControlDockWidget->setWidget( new WatchdogControl(this) );
pixhawk's avatar
pixhawk committed
521
        watchdogControlDockWidget->setObjectName("WATCHDOG_CONTROL_DOCKWIDGET");
522
        addTool(watchdogControlDockWidget, tr("Process Control"), Qt::BottomDockWidgetArea);
523
    }
524

525
    if (!hsiDockWidget) {
526 527
        hsiDockWidget = new QDockWidget(tr("Horizontal Situation Indicator"), this);
        hsiDockWidget->setWidget( new HSIDisplay(this) );
528
        hsiDockWidget->setObjectName("HORIZONTAL_SITUATION_INDICATOR_DOCK_WIDGET");
529
        addTool(hsiDockWidget, tr("Horizontal Situation"), Qt::BottomDockWidgetArea);
530
    }
531

532
    if (!headDown1DockWidget) {
533 534
        headDown1DockWidget = new QDockWidget(tr("Flight Display"), this);
        headDown1DockWidget->setWidget( new HDDisplay(acceptList, "Flight Display", this) );
535
        headDown1DockWidget->setObjectName("HEAD_DOWN_DISPLAY_1_DOCK_WIDGET");
536
        addTool(headDown1DockWidget, tr("Flight Display"), Qt::RightDockWidgetArea);
537
    }
538

539
    if (!headDown2DockWidget) {
540 541
        headDown2DockWidget = new QDockWidget(tr("Actuator Status"), this);
        headDown2DockWidget->setWidget( new HDDisplay(acceptList2, "Actuator Status", this) );
542
        headDown2DockWidget->setObjectName("HEAD_DOWN_DISPLAY_2_DOCK_WIDGET");
543
        addTool(headDown2DockWidget, tr("Actuator Status"), Qt::RightDockWidgetArea);
544
    }
545

546
    if (!rcViewDockWidget) {
547 548
        rcViewDockWidget = new QDockWidget(tr("Radio Control"), this);
        rcViewDockWidget->setWidget( new QGCRemoteControlView(this) );
549
        rcViewDockWidget->setObjectName("RADIO_CONTROL_CHANNELS_DOCK_WIDGET");
550
        addTool(rcViewDockWidget, tr("Radio Control"), Qt::BottomDockWidgetArea);
551
    }
552

553
    if (!headUpDockWidget) {
554 555
        headUpDockWidget = new QDockWidget(tr("HUD"), this);
        headUpDockWidget->setWidget( new HUD(320, 240, this));
556
        headUpDockWidget->setObjectName("HEAD_UP_DISPLAY_DOCK_WIDGET");
557
        addTool(headUpDockWidget, tr("Head Up Display"), Qt::RightDockWidgetArea);
558
    }
559

560
    if (!video1DockWidget) {
pixhawk's avatar
pixhawk committed
561 562 563 564 565 566 567
        video1DockWidget = new QDockWidget(tr("Video Stream 1"), this);
        HUD* video1 =  new HUD(160, 120, this);
        video1->enableHUDInstruments(false);
        video1->enableVideo(true);
        // FIXME select video stream as well
        video1DockWidget->setWidget(video1);
        video1DockWidget->setObjectName("VIDEO_STREAM_1_DOCK_WIDGET");
568
        addTool(video1DockWidget, tr("Video Stream 1"), Qt::LeftDockWidgetArea);
pixhawk's avatar
pixhawk committed
569 570
    }

571
    if (!video2DockWidget) {
pixhawk's avatar
pixhawk committed
572 573 574 575 576 577 578
        video2DockWidget = new QDockWidget(tr("Video Stream 2"), this);
        HUD* video2 =  new HUD(160, 120, this);
        video2->enableHUDInstruments(false);
        video2->enableVideo(true);
        // FIXME select video stream as well
        video2DockWidget->setWidget(video2);
        video2DockWidget->setObjectName("VIDEO_STREAM_2_DOCK_WIDGET");
579
        addTool(video2DockWidget, tr("Video Stream 2"), Qt::LeftDockWidgetArea);
580
    }
581

LM's avatar
LM committed
582 583 584
    // Custom widgets, added last to all menus and layouts
    buildCustomWidget();

pixhawk's avatar
pixhawk committed
585 586
    // Dialogue widgets
    //FIXME: free memory in destructor
587

588
}
589

590
void MainWindow::addTool(QDockWidget* widget, const QString& title, Qt::DockWidgetArea area)
591
{
592
    QAction* tempAction = ui.menuTools->addAction(title);
593

594
    tempAction->setCheckable(true);
595 596 597 598
    QVariant var;
    var.setValue((QWidget*)widget);
    tempAction->setData(var);
    connect(tempAction,SIGNAL(triggered(bool)),this, SLOT(showTool(bool)));
LM's avatar
LM committed
599 600
    connect(widget, SIGNAL(visibilityChanged(bool)), tempAction, SLOT(setChecked(bool)));
    tempAction->setChecked(widget->isVisible());
601
    addDockWidget(area, widget);
602 603 604
}


605
void MainWindow::showTool(bool show)
lm's avatar
lm committed
606
{
607 608 609
    QAction* act = qobject_cast<QAction *>(sender());
    QWidget* widget = qVariantValue<QWidget *>(act->data());
    widget->setVisible(show);
610
}
611

612
void MainWindow::addCentralWidget(QWidget* widget, const QString& title)
lm's avatar
lm committed
613
{
614 615 616 617
    // Check if this widget already has been added
    if (centerStack->indexOf(widget) == -1)
    {
        centerStack->addWidget(widget);
618

619
        QAction* tempAction = ui.menuMain->addAction(title);
620

621 622 623 624 625 626
        tempAction->setCheckable(true);
        QVariant var;
        var.setValue((QWidget*)widget);
        tempAction->setData(var);
        centerStackActionGroup.addAction(tempAction);
        connect(tempAction,SIGNAL(triggered()),this, SLOT(showCentralWidget()));
LM's avatar
LM committed
627 628
        connect(widget, SIGNAL(visibilityChanged(bool)), tempAction, SLOT(setChecked(bool)));
        tempAction->setChecked(widget->isVisible());
629 630 631 632
    }
}


633
void MainWindow::showCentralWidget()
634
{
635 636 637
    QAction* act = qobject_cast<QAction *>(sender());
    QWidget* widget = qVariantValue<QWidget *>(act->data());
    centerStack->setCurrentWidget(widget);
638 639
}

640 641
void MainWindow::closeEvent(QCloseEvent *event)
{
pixhawk's avatar
pixhawk committed
642
    if (isVisible()) storeViewState();
643
    storeSettings();
644
    aboutToCloseFlag = true;
645
    mavlink->storeSettings();
646
    UASManager::instance()->storeSettings();
647 648
    QMainWindow::closeEvent(event);
}
649 650 651 652

/**
 * Connect the signals and slots of the common window widgets
 */
653
void MainWindow::connectCommonWidgets()
654
{
655
    if (infoDockWidget && infoDockWidget->widget()) {
pixhawk's avatar
pixhawk committed
656 657 658
        connect(mavlink, SIGNAL(receiveLossChanged(int, float)),
                infoDockWidget->widget(), SLOT(updateSendLoss(int, float)));
    }
lm's avatar
lm committed
659 660 661 662 663
//    //TODO temporaly debug
//    if (slugsHilSimWidget && slugsHilSimWidget->widget()) {
//        connect(UASManager::instance(), SIGNAL(activeUASSet(UASInterface*)),
//                slugsHilSimWidget->widget(), SLOT(activeUasSet(UASInterface*)));
//    }
664
}
665

666 667
void MainWindow::createCustomWidget()
{
668 669
    QDockWidget* dock = new QDockWidget("Unnamed Tool", this);
    QGCToolWidget* tool = new QGCToolWidget("Unnamed Tool", dock);
lm's avatar
lm committed
670

671
    if (QGCToolWidget::instances()->size() < 2) {
lm's avatar
lm committed
672 673 674 675 676
        // This is the first widget
        ui.menuTools->addSeparator();
    }

    connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
677
    dock->setWidget(tool);
678

679 680 681 682 683 684 685 686 687 688 689 690 691 692
    QAction* showAction = new QAction(tool->getTitle(), this);
    showAction->setCheckable(true);
    connect(dock, SIGNAL(visibilityChanged(bool)), showAction, SLOT(setChecked(bool)));
    connect(showAction, SIGNAL(triggered(bool)), dock, SLOT(setVisible(bool)));
    tool->setMainMenuAction(showAction);
    ui.menuTools->addAction(showAction);
    this->addDockWidget(Qt::BottomDockWidgetArea, dock);
    dock->setVisible(true);
}

void MainWindow::loadCustomWidget()
{
    QString widgetFileExtension(".qgw");
    QString fileName = QFileDialog::getOpenFileName(this, tr("Specify Widget File Name"), QDesktopServices::storageLocation(QDesktopServices::DesktopLocation), tr("QGroundControl Widget (*%1);;").arg(widgetFileExtension));
693
    if (fileName != "") loadCustomWidget(fileName);
LM's avatar
LM committed
694
}
695

LM's avatar
LM committed
696 697 698 699
void MainWindow::loadCustomWidget(const QString& fileName, bool singleinstance)
{
    QGCToolWidget* tool = new QGCToolWidget("", this);
    if (tool->loadSettings(fileName, true) || !singleinstance)
700
    {
LM's avatar
LM committed
701 702 703 704
        // Add widget to UI
        QDockWidget* dock = new QDockWidget(tool->getTitle(), this);
        connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
        dock->setWidget(tool);
705
//        dock->setObjectName();
LM's avatar
LM committed
706 707
        tool->setParent(dock);

708
        QAction* showAction = new QAction(tool->getTitle(), this);
LM's avatar
LM committed
709 710 711 712 713 714 715 716 717 718 719
        showAction->setCheckable(true);
        connect(dock, SIGNAL(visibilityChanged(bool)), showAction, SLOT(setChecked(bool)));
        connect(showAction, SIGNAL(triggered(bool)), dock, SLOT(setVisible(bool)));
        tool->setMainMenuAction(showAction);
        ui.menuTools->addAction(showAction);
        this->addDockWidget(Qt::BottomDockWidgetArea, dock);
        dock->setVisible(true);
    }
    else
    {
        return;
720
    }
721 722
}

LM's avatar
LM committed
723
void MainWindow::loadCustomWidgetsFromDefaults(const QString& systemType, const QString& autopilotType)
724
{
LM's avatar
LM committed
725
    QString defaultsDir = qApp->applicationDirPath() + "/files/" + systemType.toLower() + "/" + autopilotType.toLower() + "/widgets/";
726

LM's avatar
LM committed
727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742
    QDir widgets(defaultsDir);
    QStringList files = widgets.entryList();
    if (files.count() == 0)
    {
        qDebug() << "No default custom widgets for system " << systemType << "autopilot" << autopilotType << " found";
        qDebug() << "Tried with path: " << defaultsDir;
    }

    // Load all custom widgets found in the AP folder
    for(int i = 0; i < files.count(); ++i)
    {
        QString file = files[i];
        if (file.endsWith(".qgw"))
        {
            // Will only be loaded if not already a custom widget with
            // the same name is present
743 744
            qDebug() << "Tried to load file: " << defaultsDir+file;
            loadCustomWidget(defaultsDir+file, true);
LM's avatar
LM committed
745 746
        }
    }
747 748
}

749 750 751 752 753 754
void MainWindow::loadSettings()
{
    QSettings settings;
    settings.beginGroup("QGC_MAINWINDOW");
    autoReconnect = settings.value("AUTO_RECONNECT", autoReconnect).toBool();
    currentStyle = (QGC_MAINWINDOW_STYLE)settings.value("CURRENT_STYLE", currentStyle).toInt();
755
    lowPowerMode = settings.value("LOW_POWER_MODE", lowPowerMode).toBool();
756 757 758 759 760 761 762 763 764 765
    settings.endGroup();
}

void MainWindow::storeSettings()
{
    QSettings settings;
    settings.beginGroup("QGC_MAINWINDOW");
    settings.setValue("AUTO_RECONNECT", autoReconnect);
    settings.setValue("CURRENT_STYLE", currentStyle);
    settings.endGroup();
pixhawk's avatar
pixhawk committed
766 767 768 769 770 771 772 773 774 775 776
    if (!aboutToCloseFlag && isVisible())
    {
        settings.setValue(getWindowGeometryKey(), saveGeometry());
        // Save the last current view in any case
        settings.setValue("CURRENT_VIEW", currentView);
        // Save the current window state, but only if a system is connected (else no real number of widgets would be present)
        if (UASManager::instance()->getUASList().length() > 0) settings.setValue(getWindowStateKey(), saveState(QGC::applicationVersion()));
        // Save the current view only if a UAS is connected
        if (UASManager::instance()->getUASList().length() > 0) settings.setValue("CURRENT_VIEW_WITH_UAS_CONNECTED", currentView);
        // Save the current power mode
    }
777
    settings.setValue("LOW_POWER_MODE", lowPowerMode);
778 779 780
    settings.sync();
}

781 782
void MainWindow::configureWindowName()
{
pixhawk's avatar
pixhawk committed
783 784 785
    QList<QHostAddress> hostAddresses = QNetworkInterface::allAddresses();
    QString windowname = qApp->applicationName() + " " + qApp->applicationVersion();
    bool prevAddr = false;
786

pixhawk's avatar
pixhawk committed
787
    windowname.append(" (" + QHostInfo::localHostName() + ": ");
788

789 790
    for (int i = 0; i < hostAddresses.size(); i++)
    {
pixhawk's avatar
pixhawk committed
791
        // Exclude loopback IPv4 and all IPv6 addresses
792 793
        if (hostAddresses.at(i) != QHostAddress("127.0.0.1") && !hostAddresses.at(i).toString().contains(":"))
        {
pixhawk's avatar
pixhawk committed
794 795 796 797 798
            if(prevAddr) windowname.append("/");
            windowname.append(hostAddresses.at(i).toString());
            prevAddr = true;
        }
    }
799

pixhawk's avatar
pixhawk committed
800
    windowname.append(")");
801

pixhawk's avatar
pixhawk committed
802
    setWindowTitle(windowname);
803 804

#ifndef Q_WS_MAC
pixhawk's avatar
pixhawk committed
805
    //qApp->setWindowIcon(QIcon(":/core/images/qtcreator_logo_128.png"));
806 807 808
#endif
}

pixhawk's avatar
pixhawk committed
809
void MainWindow::startVideoCapture()
pixhawk's avatar
pixhawk committed
810 811 812 813 814
{
    QString format = "bmp";
    QString initialPath = QDir::currentPath() + tr("/untitled.") + format;

    QString screenFileName = QFileDialog::getSaveFileName(this, tr("Save As"),
815 816 817 818
                             initialPath,
                             tr("%1 Files (*.%2);;All Files (*)")
                             .arg(format.toUpper())
                             .arg(format));
pixhawk's avatar
pixhawk committed
819 820
    delete videoTimer;
    videoTimer = new QTimer(this);
821 822 823
    //videoTimer->setInterval(40);
    //connect(videoTimer, SIGNAL(timeout()), this, SLOT(saveScreen()));
    //videoTimer->stop();
pixhawk's avatar
pixhawk committed
824 825
}

pixhawk's avatar
pixhawk committed
826
void MainWindow::stopVideoCapture()
pixhawk's avatar
pixhawk committed
827 828 829 830 831 832
{
    videoTimer->stop();

    // TODO Convert raw images to PNG
}

pixhawk's avatar
pixhawk committed
833
void MainWindow::saveScreen()
pixhawk's avatar
pixhawk committed
834 835 836 837
{
    QPixmap window = QPixmap::grabWindow(this->winId());
    QString format = "bmp";

838 839
    if (!screenFileName.isEmpty())
    {
pixhawk's avatar
pixhawk committed
840 841 842 843
        window.save(screenFileName, format.toAscii());
    }
}

844
void MainWindow::enableAutoReconnect(bool enabled)
pixhawk's avatar
pixhawk committed
845
{
846 847
    autoReconnect = enabled;
}
848

849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865
void MainWindow::loadNativeStyle()
{
    loadStyle(QGC_MAINWINDOW_STYLE_NATIVE);
}

void MainWindow::loadIndoorStyle()
{
    loadStyle(QGC_MAINWINDOW_STYLE_INDOOR);
}

void MainWindow::loadOutdoorStyle()
{
    loadStyle(QGC_MAINWINDOW_STYLE_OUTDOOR);
}

void MainWindow::loadStyle(QGC_MAINWINDOW_STYLE style)
{
866 867 868 869 870 871 872 873 874
    switch (style) {
    case QGC_MAINWINDOW_STYLE_NATIVE: {
        // Native mode means setting no style
        // so if we were already in native mode
        // take no action
        // Only if a style was set, remove it.
        if (style != currentStyle) {
            qApp->setStyleSheet("");
            showInfoMessage(tr("Please restart QGroundControl"), tr("Please restart QGroundControl to switch to fully native look and feel. Currently you have loaded Qt's plastique style."));
875
        }
876 877
    }
    break;
878 879 880 881 882 883 884 885 886 887 888 889 890 891
    case QGC_MAINWINDOW_STYLE_INDOOR:
        qApp->setStyle("plastique");
        styleFileName = ":/images/style-mission.css";
        reloadStylesheet();
        break;
    case QGC_MAINWINDOW_STYLE_OUTDOOR:
        qApp->setStyle("plastique");
        styleFileName = ":/images/style-outdoor.css";
        reloadStylesheet();
        break;
    }
    currentStyle = style;
}

892
void MainWindow::selectStylesheet()
pixhawk's avatar
pixhawk committed
893
{
894
    // Let user select style sheet
895
    styleFileName = QFileDialog::getOpenFileName(this, tr("Specify stylesheet"), styleFileName, tr("CSS Stylesheet (*.css);;"));
896

897
    if (!styleFileName.endsWith(".css")) {
898 899 900 901 902 903 904 905 906 907
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Information);
        msgBox.setText(tr("QGroundControl did lot load a new style"));
        msgBox.setInformativeText(tr("No suitable .css file selected. Please select a valid .css file."));
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
        return;
    }

pixhawk's avatar
pixhawk committed
908
    // Load style sheet
909 910 911 912 913 914 915
    reloadStylesheet();
}

void MainWindow::reloadStylesheet()
{
    // Load style sheet
    QFile* styleSheet = new QFile(styleFileName);
916
    if (!styleSheet->exists()) {
917 918
        styleSheet = new QFile(":/images/style-mission.css");
    }
919
    if (styleSheet->open(QIODevice::ReadOnly | QIODevice::Text)) {
920 921
        QString style = QString(styleSheet->readAll());
        style.replace("ICONDIR", QCoreApplication::applicationDirPath()+ "/images/");
pixhawk's avatar
pixhawk committed
922
        qApp->setStyleSheet(style);
923
    } else {
924 925 926
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Information);
        msgBox.setText(tr("QGroundControl did lot load a new style"));
927
        msgBox.setInformativeText(tr("Stylesheet file %1 was not readable").arg(styleFileName));
928 929 930
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
pixhawk's avatar
pixhawk committed
931
    }
932
    delete styleSheet;
pixhawk's avatar
pixhawk committed
933 934
}

935 936 937 938 939 940
/**
 * The status message will be overwritten if a new message is posted to this function
 *
 * @param status message text
 * @param timeout how long the status should be displayed
 */
pixhawk's avatar
pixhawk committed
941
void MainWindow::showStatusMessage(const QString& status, int timeout)
pixhawk's avatar
pixhawk committed
942
{
lm's avatar
lm committed
943
    statusBar()->showMessage(status, timeout);
pixhawk's avatar
pixhawk committed
944 945
}

946 947 948 949 950 951
/**
 * The status message will be overwritten if a new message is posted to this function.
 * it will be automatically hidden after 5 seconds.
 *
 * @param status message text
 */
952
void MainWindow::showStatusMessage(const QString& status)
pixhawk's avatar
pixhawk committed
953
{
lm's avatar
lm committed
954
    statusBar()->showMessage(status, 20000);
955 956 957 958
}

void MainWindow::showCriticalMessage(const QString& title, const QString& message)
{
959
    QMessageBox msgBox(this);
960 961 962 963 964 965
    msgBox.setIcon(QMessageBox::Critical);
    msgBox.setText(title);
    msgBox.setInformativeText(message);
    msgBox.setStandardButtons(QMessageBox::Ok);
    msgBox.setDefaultButton(QMessageBox::Ok);
    msgBox.exec();
pixhawk's avatar
pixhawk committed
966 967
}

lm's avatar
lm committed
968 969
void MainWindow::showInfoMessage(const QString& title, const QString& message)
{
970
    QMessageBox msgBox(this);
lm's avatar
lm committed
971 972 973 974 975 976 977 978
    msgBox.setIcon(QMessageBox::Information);
    msgBox.setText(title);
    msgBox.setInformativeText(message);
    msgBox.setStandardButtons(QMessageBox::Ok);
    msgBox.setDefaultButton(QMessageBox::Ok);
    msgBox.exec();
}

pixhawk's avatar
pixhawk committed
979 980 981 982
/**
* @brief Create all actions associated to the main window
*
**/
983
void MainWindow::connectCommonActions()
pixhawk's avatar
pixhawk committed
984
{
985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007
    // Bind together the perspective actions
    QActionGroup* perspectives = new QActionGroup(ui.menuPerspectives);
    perspectives->addAction(ui.actionEngineersView);
    perspectives->addAction(ui.actionMavlinkView);
    perspectives->addAction(ui.actionPilotsView);
    perspectives->addAction(ui.actionOperatorsView);
    perspectives->addAction(ui.actionUnconnectedView);
    perspectives->setExclusive(true);

    // Mark the right one as selected
    if (currentView == VIEW_ENGINEER) ui.actionEngineersView->setChecked(true);
    if (currentView == VIEW_MAVLINK) ui.actionMavlinkView->setChecked(true);
    if (currentView == VIEW_PILOT) ui.actionPilotsView->setChecked(true);
    if (currentView == VIEW_OPERATOR) ui.actionOperatorsView->setChecked(true);
    if (currentView == VIEW_UNCONNECTED) ui.actionUnconnectedView->setChecked(true);

    // The UAS actions are not enabled without connection to system
    ui.actionLiftoff->setEnabled(false);
    ui.actionLand->setEnabled(false);
    ui.actionEmergency_Kill->setEnabled(false);
    ui.actionEmergency_Land->setEnabled(false);
    ui.actionShutdownMAV->setEnabled(false);

pixhawk's avatar
pixhawk committed
1008 1009 1010 1011 1012
    // Connect actions from ui
    connect(ui.actionAdd_Link, SIGNAL(triggered()), this, SLOT(addLink()));

    // Connect internal actions
    connect(UASManager::instance(), SIGNAL(UASCreated(UASInterface*)), this, SLOT(UASCreated(UASInterface*)));
1013
    connect(UASManager::instance(), SIGNAL(activeUASSet(UASInterface*)), this, SLOT(setActiveUAS(UASInterface*)));
pixhawk's avatar
pixhawk committed
1014

1015
    // Unmanned System controls
pixhawk's avatar
pixhawk committed
1016 1017 1018 1019
    connect(ui.actionLiftoff, SIGNAL(triggered()), UASManager::instance(), SLOT(launchActiveUAS()));
    connect(ui.actionLand, SIGNAL(triggered()), UASManager::instance(), SLOT(returnActiveUAS()));
    connect(ui.actionEmergency_Land, SIGNAL(triggered()), UASManager::instance(), SLOT(stopActiveUAS()));
    connect(ui.actionEmergency_Kill, SIGNAL(triggered()), UASManager::instance(), SLOT(killActiveUAS()));
1020
    connect(ui.actionShutdownMAV, SIGNAL(triggered()), UASManager::instance(), SLOT(shutdownActiveUAS()));
pixhawk's avatar
pixhawk committed
1021 1022
    connect(ui.actionConfiguration, SIGNAL(triggered()), UASManager::instance(), SLOT(configureActiveUAS()));

1023
    // Views actions
1024
    connect(ui.actionPilotsView, SIGNAL(triggered()), this, SLOT(loadPilotView()));
1025
    connect(ui.actionEngineersView, SIGNAL(triggered()), this, SLOT(loadEngineerView()));
1026
    connect(ui.actionOperatorsView, SIGNAL(triggered()), this, SLOT(loadOperatorView()));
1027
    connect(ui.actionUnconnectedView, SIGNAL(triggered()), this, SLOT(loadUnconnectedView()));
1028

1029
    connect(ui.actionMavlinkView, SIGNAL(triggered()), this, SLOT(loadMAVLinkView()));
1030 1031
    connect(ui.actionReloadStylesheet, SIGNAL(triggered()), this, SLOT(reloadStylesheet()));
    connect(ui.actionSelectStylesheet, SIGNAL(triggered()), this, SLOT(selectStylesheet()));
1032

1033
    // Help Actions
1034
    connect(ui.actionOnline_Documentation, SIGNAL(triggered()), this, SLOT(showHelp()));
1035
    connect(ui.actionDeveloper_Credits, SIGNAL(triggered()), this, SLOT(showCredits()));
1036
    connect(ui.actionProject_Roadmap_2, SIGNAL(triggered()), this, SLOT(showRoadMap()));
1037 1038 1039

    // Custom widget actions
    connect(ui.actionNewCustomWidget, SIGNAL(triggered()), this, SLOT(createCustomWidget()));
1040
    connect(ui.actionLoadCustomWidgetFile, SIGNAL(triggered()), this, SLOT(loadCustomWidget()));
1041 1042 1043

    // Audio output
    ui.actionMuteAudioOutput->setChecked(GAudioOutput::instance()->isMuted());
1044
    connect(GAudioOutput::instance(), SIGNAL(mutedChanged(bool)), ui.actionMuteAudioOutput, SLOT(setChecked(bool)));
1045
    connect(ui.actionMuteAudioOutput, SIGNAL(triggered(bool)), GAudioOutput::instance(), SLOT(mute(bool)));
1046 1047 1048 1049 1050 1051 1052

    // User interaction
    // NOTE: Joystick thread is not started and
    // configuration widget is not instantiated
    // unless it is actually used
    // so no ressources spend on this.
    ui.actionJoystickSettings->setVisible(true);
1053 1054 1055

    // Configuration
    // Joystick
1056
    connect(ui.actionJoystickSettings, SIGNAL(triggered()), this, SLOT(configure()));
1057 1058
    // Application Settings
    connect(ui.actionSettings, SIGNAL(triggered()), this, SLOT(showSettings()));
1059 1060
}

1061 1062
void MainWindow::showHelp()
{
1063
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/users/start"))) {
1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Critical);
        msgBox.setText("Could not open help in browser");
        msgBox.setInformativeText("To get to the online help, please open http://qgroundcontrol.org/user_guide in a browser.");
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
    }
}

void MainWindow::showCredits()
{
1076
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/credits"))) {
1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Critical);
        msgBox.setText("Could not open credits in browser");
        msgBox.setInformativeText("To get to the online help, please open http://qgroundcontrol.org/credits in a browser.");
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
    }
}

void MainWindow::showRoadMap()
{
1089
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/dev/roadmap"))) {
1090 1091 1092 1093 1094 1095 1096 1097 1098 1099
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Critical);
        msgBox.setText("Could not open roadmap in browser");
        msgBox.setInformativeText("To get to the online help, please open http://qgroundcontrol.org/roadmap in a browser.");
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
    }
}

pixhawk's avatar
pixhawk committed
1100
void MainWindow::configure()
pixhawk's avatar
pixhawk committed
1101
{
1102 1103
    if (!joystickWidget) {
        if (!joystick->isRunning()) {
1104 1105 1106 1107 1108
            joystick->start();
        }
        joystickWidget = new JoystickWidget(joystick);
    }
    joystickWidget->show();
pixhawk's avatar
pixhawk committed
1109 1110
}

1111 1112 1113 1114 1115 1116
void MainWindow::showSettings()
{
    QGCSettingsWidget* settings = new QGCSettingsWidget(this);
    settings->show();
}

pixhawk's avatar
pixhawk committed
1117
void MainWindow::addLink()
pixhawk's avatar
pixhawk committed
1118 1119 1120 1121
{
    SerialLink* link = new SerialLink();
    // TODO This should be only done in the dialog itself

1122
    LinkManager::instance()->add(link);
pixhawk's avatar
pixhawk committed
1123 1124
    LinkManager::instance()->addProtocol(link, mavlink);

1125 1126
    // Go fishing for this link's configuration window
    QList<QAction*> actions = ui.menuNetwork->actions();
pixhawk's avatar
pixhawk committed
1127

1128 1129
    foreach (QAction* act, actions) {
        if (act->data().toInt() == LinkManager::instance()->getLinks().indexOf(link)) {
1130 1131 1132 1133
            act->trigger();
            break;
        }
    }
pixhawk's avatar
pixhawk committed
1134 1135
}

pixhawk's avatar
pixhawk committed
1136 1137
void MainWindow::addLink(LinkInterface *link)
{
1138 1139 1140 1141
    // IMPORTANT! KEEP THESE TWO LINES
    // THEY MAKE SURE THE LINK IS PROPERLY REGISTERED
    // BEFORE LINKING THE UI AGAINST IT
    // Register (does nothing if already registered)
1142
    LinkManager::instance()->add(link);
1143
    LinkManager::instance()->addProtocol(link, mavlink);
1144

1145 1146
    // Go fishing for this link's configuration window
    QList<QAction*> actions = ui.menuNetwork->actions();
pixhawk's avatar
pixhawk committed
1147

1148
    bool found = false;
1149

1150 1151
    foreach (QAction* act, actions) {
        if (act->data().toInt() == LinkManager::instance()->getLinks().indexOf(link)) {
1152 1153 1154 1155 1156
            found = true;
        }
    }

    UDPLink* udp = dynamic_cast<UDPLink*>(link);
1157

1158
    if (!found || udp) {
1159 1160 1161 1162 1163 1164 1165 1166
        CommConfigurationWindow* commWidget = new CommConfigurationWindow(link, mavlink, this);
        QAction* action = commWidget->getAction();
        ui.menuNetwork->addAction(action);

        // Error handling
        connect(link, SIGNAL(communicationError(QString,QString)), this, SLOT(showCriticalMessage(QString,QString)), Qt::QueuedConnection);
        // Special case for simulationlink
        MAVLinkSimulationLink* sim = dynamic_cast<MAVLinkSimulationLink*>(link);
1167
        if (sim) {
1168 1169 1170
            //connect(sim, SIGNAL(valueChanged(int,QString,double,quint64)), linechart, SLOT(appendData(int,QString,double,quint64)));
            connect(ui.actionSimulate, SIGNAL(triggered(bool)), sim, SLOT(connectLink(bool)));
        }
pixhawk's avatar
pixhawk committed
1171 1172 1173
    }
}

1174 1175 1176 1177 1178 1179 1180
void MainWindow::setActiveUAS(UASInterface* uas)
{
    // Enable and rename menu
    ui.menuUnmanned_System->setTitle(uas->getUASName());
    if (!ui.menuUnmanned_System->isEnabled()) ui.menuUnmanned_System->setEnabled(true);
}

1181 1182 1183
void MainWindow::UASSpecsChanged(int uas)
{
    UASInterface* activeUAS = UASManager::instance()->getActiveUAS();
1184 1185
    if (activeUAS) {
        if (activeUAS->getUASID() == uas) {
1186 1187 1188 1189 1190
            ui.menuUnmanned_System->setTitle(activeUAS->getUASName());
        }
    }
}

pixhawk's avatar
pixhawk committed
1191
void MainWindow::UASCreated(UASInterface* uas)
pixhawk's avatar
pixhawk committed
1192
{
1193

pixhawk's avatar
pixhawk committed
1194 1195
    // Connect the UAS to the full user interface

LM's avatar
LM committed
1196 1197
    if (uas != NULL)
    {
1198
        // The pilot, operator and engineer views were not available on startup, enable them now
1199
        ui.actionPilotsView->setEnabled(true);
1200 1201
        ui.actionOperatorsView->setEnabled(true);
        ui.actionEngineersView->setEnabled(true);
1202 1203 1204 1205 1206 1207
        // The UAS actions are not enabled without connection to system
        ui.actionLiftoff->setEnabled(true);
        ui.actionLand->setEnabled(true);
        ui.actionEmergency_Kill->setEnabled(true);
        ui.actionEmergency_Land->setEnabled(true);
        ui.actionShutdownMAV->setEnabled(true);
1208

1209 1210
        QIcon icon;
        // Set matching icon
1211
        switch (uas->getSystemType()) {
1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234
        case 0:
            icon = QIcon(":/images/mavs/generic.svg");
            break;
        case 1:
            icon = QIcon(":/images/mavs/fixed-wing.svg");
            break;
        case 2:
            icon = QIcon(":/images/mavs/quadrotor.svg");
            break;
        case 3:
            icon = QIcon(":/images/mavs/coaxial.svg");
            break;
        case 4:
            icon = QIcon(":/images/mavs/helicopter.svg");
            break;
        case 5:
            icon = QIcon(":/images/mavs/groundstation.svg");
            break;
        default:
            icon = QIcon(":/images/mavs/unknown.svg");
            break;
        }

1235 1236 1237
        QAction* uasAction = new QAction(icon, tr("Select %1 for control").arg(uas->getUASName()), ui.menuConnected_Systems);
        connect(uas, SIGNAL(systemRemoved()), uasAction, SLOT(deleteLater()));
        connect(uasAction, SIGNAL(triggered()), uas, SLOT(setSelected()));
1238
        connect(uas, SIGNAL(systemSpecsChanged(int)), this, SLOT(UASSpecsChanged(int)));
1239 1240

        ui.menuConnected_Systems->addAction(uasAction);
1241 1242

        // FIXME Should be not inside the mainwindow
1243
        if (debugConsoleDockWidget) {
pixhawk's avatar
pixhawk committed
1244
            DebugConsole *debugConsole = dynamic_cast<DebugConsole*>(debugConsoleDockWidget->widget());
1245
            if (debugConsole) {
pixhawk's avatar
pixhawk committed
1246 1247 1248
                connect(uas, SIGNAL(textMessageReceived(int,int,int,QString)),
                        debugConsole, SLOT(receiveTextMessage(int,int,int,QString)));
            }
1249
        }
1250 1251

        // Health / System status indicator
1252
        if (infoDockWidget) {
pixhawk's avatar
pixhawk committed
1253
            UASInfoWidget *infoWidget = dynamic_cast<UASInfoWidget*>(infoDockWidget->widget());
1254
            if (infoWidget) {
pixhawk's avatar
pixhawk committed
1255 1256
                infoWidget->addUAS(uas);
            }
1257
        }
1258 1259

        // UAS List
1260
        if (listDockWidget) {
pixhawk's avatar
pixhawk committed
1261
            UASListWidget *listWidget = dynamic_cast<UASListWidget*>(listDockWidget->widget());
1262
            if (listWidget) {
pixhawk's avatar
pixhawk committed
1263 1264
                listWidget->addUAS(uas);
            }
1265
        }
1266

1267 1268 1269 1270 1271 1272
        // Line chart
        if (!linechartWidget)
        {
            // Center widgets
            linechartWidget   = new Linecharts(this);
            addCentralWidget(linechartWidget, tr("Realtime Plot"));
1273 1274
        }

LM's avatar
LM committed
1275 1276 1277 1278 1279
        // Load default custom widgets for this autopilot type
        loadCustomWidgetsFromDefaults(uas->getSystemTypeString(uas->getSystemType()), uas->getAutopilotTypeString(uas->getAutopilotType()));



1280 1281 1282 1283
        // Change the view only if this is the first UAS

        // If this is the first connected UAS, it is both created as well as
        // the currently active UAS
1284 1285
        if (UASManager::instance()->getUASList().size() == 1)
        {
1286
            // Load last view if setting is present
1287 1288
            if (settings.contains("CURRENT_VIEW_WITH_UAS_CONNECTED"))
            {
1289
                int view = settings.value("CURRENT_VIEW_WITH_UAS_CONNECTED").toInt();
1290
                switch (view) {
1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306
                case VIEW_ENGINEER:
                    loadEngineerView();
                    break;
                case VIEW_MAVLINK:
                    loadMAVLinkView();
                    break;
                case VIEW_PILOT:
                    loadPilotView();
                    break;
                case VIEW_UNCONNECTED:
                    loadUnconnectedView();
                    break;
                case VIEW_OPERATOR:
                default:
                    loadOperatorView();
                    break;
1307
                }
1308 1309 1310
            }
            else
            {
1311
                loadOperatorView();
1312
            }
1313
        }
1314

1315
    }
1316 1317

    if (!ui.menuConnected_Systems->isEnabled()) ui.menuConnected_Systems->setEnabled(true);
lm's avatar
lm committed
1318

1319
    // Restore the mainwindow size
LM's avatar
LM committed
1320 1321
    if (settings.contains(getWindowGeometryKey()))
    {
1322 1323
        restoreGeometry(settings.value(getWindowGeometryKey()).toByteArray());
    }
pixhawk's avatar
pixhawk committed
1324 1325
}

1326
/**
1327
 * Stores the current view state
1328
 */
1329
void MainWindow::storeViewState()
1330
{
pixhawk's avatar
pixhawk committed
1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341
    if (!aboutToCloseFlag)
    {
        // Save current state
        settings.setValue(getWindowStateKey(), saveState(QGC::applicationVersion()));
        settings.setValue(getWindowStateKey()+"CENTER_WIDGET", centerStack->currentIndex());
        // Although we want save the state of the window, we do not want to change the top-leve state (minimized, maximized, etc)
        // therefore this state is stored here and restored after applying the rest of the settings in the new
        // perspective.
        windowStateVal = this->windowState();
        settings.setValue(getWindowGeometryKey(), saveGeometry());
    }
1342
}
1343

1344 1345 1346 1347
void MainWindow::loadViewState()
{
    // Restore center stack state
    int index = settings.value(getWindowStateKey()+"CENTER_WIDGET", centerStack->currentIndex()).toInt();
LM's avatar
LM committed
1348 1349 1350 1351 1352 1353 1354
    // The offline plot view is usually the consequence of a logging run, always show the realtime view first
    if (centerStack->indexOf(dataplotWidget) == index)
    {
        // Rewrite to realtime plot
        index = centerStack->indexOf(linechartWidget);
    }
    if (index != -1) centerStack->setCurrentIndex(index);
pixhawk's avatar
pixhawk committed
1355

1356 1357 1358 1359
    // Restore the widget positions and size
    if (settings.contains(getWindowStateKey()))
    {
        restoreState(settings.value(getWindowStateKey()).toByteArray(), QGC::applicationVersion());
pixhawk's avatar
pixhawk committed
1360 1361 1362
    }
}

1363
void MainWindow::loadEngineerView()
lm's avatar
lm committed
1364
{
1365
    if (currentView != VIEW_ENGINEER) {
1366
        storeViewState();
1367 1368
        currentView = VIEW_ENGINEER;
        ui.actionEngineersView->setChecked(true);
1369
        loadViewState();
1370
    }
lm's avatar
lm committed
1371 1372
}

1373
void MainWindow::loadOperatorView()
lm's avatar
lm committed
1374
{
1375
    if (currentView != VIEW_OPERATOR) {
1376
        storeViewState();
1377 1378
        currentView = VIEW_OPERATOR;
        ui.actionOperatorsView->setChecked(true);
1379
        loadViewState();
1380 1381
    }
}
lm's avatar
lm committed
1382

1383 1384
void MainWindow::loadUnconnectedView()
{
1385 1386 1387
    if (currentView != VIEW_UNCONNECTED)
    {
        storeViewState();
1388 1389
        currentView = VIEW_UNCONNECTED;
        ui.actionUnconnectedView->setChecked(true);
1390
        loadViewState();
1391
    }
lm's avatar
lm committed
1392 1393
}

1394
void MainWindow::loadPilotView()
1395
{
1396 1397 1398
    if (currentView != VIEW_PILOT)
    {
        storeViewState();
1399 1400
        currentView = VIEW_PILOT;
        ui.actionPilotsView->setChecked(true);
1401
        loadViewState();
1402
    }
1403 1404
}

1405
void MainWindow::loadMAVLinkView()
1406
{
1407 1408 1409
    if (currentView != VIEW_MAVLINK)
    {
        storeViewState();
1410 1411
        currentView = VIEW_MAVLINK;
        ui.actionMavlinkView->setChecked(true);
1412
        loadViewState();
1413
    }
1414 1415
}

1416 1417
void MainWindow::loadDataView(QString fileName)
{
1418
    // Plot is now selected, now load data from file
1419 1420 1421 1422 1423 1424 1425 1426
    if (dataplotWidget)
    {
        dataplotWidget->loadFile(fileName);
    }
    QStackedWidget *centerStack = dynamic_cast<QStackedWidget*>(centralWidget());
    if (centerStack)
    {
        centerStack->setCurrentWidget(dataplotWidget);
1427
        dataplotWidget->loadFile(fileName);
1428
    }
1429
}