MainWindow.cc 51.5 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
 */

#include <QSettings>
#include <QDockWidget>
#include <QNetworkInterface>
#include <QMessageBox>
#include <QDebug>
#include <QTimer>
#include <QHostInfo>
lm's avatar
lm committed
37
#include <QSplashScreen>
pixhawk's avatar
pixhawk committed
38

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

56
#ifdef QGC_OSG_ENABLED
57
#include "Q3DWidgetFactory.h"
58
#endif
pixhawk's avatar
pixhawk committed
59

lm's avatar
lm committed
60 61 62 63
// FIXME Move
#include "PxQuadMAV.h"
#include "SlugsMAV.h"

pixhawk's avatar
pixhawk committed
64

65
#include "LogCompressor.h"
pixhawk's avatar
pixhawk committed
66

lm's avatar
lm committed
67
MainWindow* MainWindow::instance(QSplashScreen* screen)
68
{
69
    static MainWindow* _instance = 0;
70 71
    if(_instance == 0)
    {
72
        _instance = new MainWindow();
lm's avatar
lm committed
73
        if (screen) connect(_instance, SIGNAL(initStatusChanged(QString)), screen, SLOT(showMessage(QString)));
74

75
        /* Set the application as parent to ensure that this object
76
                 * will be destroyed when the main application exits */
77 78 79
        //_instance->setParent(qApp);
    }
    return _instance;
80 81
}

pixhawk's avatar
pixhawk committed
82 83 84 85 86 87 88
/**
* 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()
**/
89
MainWindow::MainWindow(QWidget *parent):
90 91
    QMainWindow(parent),
    currentView(VIEW_UNCONNECTED),
92
    currentStyle(QGC_MAINWINDOW_STYLE_INDOOR),
93 94
    aboutToCloseFlag(false),
    changingViewsFlag(false),
95
    centerStackActionGroup(new QActionGroup(this)),
96
    styleFileName(QCoreApplication::applicationDirPath() + "/style-indoor.css"),
97
    autoReconnect(false),
98
    lowPowerMode(false)
pixhawk's avatar
pixhawk committed
99
{
lm's avatar
lm committed
100 101
    hide();
    emit initStatusChanged("Loading UI Settings..");
102
    loadSettings();
103 104
    if (!settings.contains("CURRENT_VIEW"))
    {
105 106
        // Set this view as default view
        settings.setValue("CURRENT_VIEW", currentView);
107 108 109
    }
    else
    {
110 111 112
        // LOAD THE LAST VIEW
        VIEW_SECTIONS currentViewCandidate = (VIEW_SECTIONS) settings.value("CURRENT_VIEW", currentView).toInt();
        if (currentViewCandidate != VIEW_ENGINEER &&
113
                currentViewCandidate != VIEW_OPERATOR &&
114 115 116
                currentViewCandidate != VIEW_PILOT &&
                currentViewCandidate != VIEW_FULL)
        {
117
            currentView = currentViewCandidate;
118
        }
119 120
    }

121 122
    settings.sync();

lm's avatar
lm committed
123
    emit initStatusChanged("Loading Style.");
124
    loadStyle(currentStyle);
125

lm's avatar
lm committed
126 127
    emit initStatusChanged("Setting up user interface.");

pixhawk's avatar
pixhawk committed
128 129
    // Setup user interface
    ui.setupUi(this);
lm's avatar
lm committed
130
    hide();
pixhawk's avatar
pixhawk committed
131

132 133 134 135 136 137
    // Set dock options
    setDockOptions(AnimatedDocks | AllowTabbedDocks | AllowNestedDocks);
    statusBar()->setSizeGripEnabled(true);

    configureWindowName();

138 139 140
    // Setup corners
    setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);

141
    // Setup UI state machines
142
	centerStackActionGroup->setExclusive(true);
143

144 145
    centerStack = new QStackedWidget(this);
    setCentralWidget(centerStack);
146

lm's avatar
lm committed
147 148 149 150 151 152 153 154
    // Load Toolbar
    toolBar = new QGCToolBar(this);
    this->addToolBar(toolBar);
    // Add actions
    toolBar->addPerspectiveChangeAction(ui.actionOperatorsView);
    toolBar->addPerspectiveChangeAction(ui.actionEngineersView);
    toolBar->addPerspectiveChangeAction(ui.actionPilotsView);

lm's avatar
lm committed
155 156
    emit initStatusChanged("Building common widgets.");

157 158
    buildCommonWidgets();
    connectCommonWidgets();
159

lm's avatar
lm committed
160 161
    emit initStatusChanged("Building common actions.");

162
    // Create actions
163
    connectCommonActions();
164

165
    // Populate link menu
lm's avatar
lm committed
166
    emit initStatusChanged("Populating link menu");
167
    QList<LinkInterface*> links = LinkManager::instance()->getLinks();
168 169
    foreach(LinkInterface* link, links)
    {
170 171
        this->addLink(link);
    }
172

173 174
    connect(LinkManager::instance(), SIGNAL(newLink(LinkInterface*)), this, SLOT(addLink(LinkInterface*)));

175
    // Connect user interface devices
lm's avatar
lm committed
176
    emit initStatusChanged("Initializing joystick interface.");
177 178
    joystickWidget = 0;
    joystick = new JoystickInput();
179

180
    // Connect link
181 182
    if (autoReconnect)
    {
183 184 185 186 187 188
        SerialLink* link = new SerialLink();
        // Add to registry
        LinkManager::instance()->add(link);
        LinkManager::instance()->addProtocol(link, mavlink);
        link->connect();
    }
189

190 191 192
    // Set low power mode
    enableLowPowerMode(lowPowerMode);

193 194
    // Initialize window state
    windowStateVal = windowState();
195

lm's avatar
lm committed
196 197
    emit initStatusChanged("Restoring last view state.");

198 199 200
    // Restore the window setup
    loadViewState();

lm's avatar
lm committed
201
    emit initStatusChanged("Restoring last window size.");
202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
    // Restore the window position and size
    if (settings.contains(getWindowGeometryKey()))
    {
        // Restore the window geometry
        restoreGeometry(settings.value(getWindowGeometryKey()).toByteArray());
        show();
    }
    else
    {
        // Adjust the size
        const int screenWidth = QApplication::desktop()->width();
        const int screenHeight = QApplication::desktop()->height();

        if (screenWidth < 1200)
        {
            showFullScreen();
        }
        else
        {
            resize(screenWidth*0.67f, qMin(screenHeight, (int)(screenWidth*0.67f*0.67f)));
            show();
        }

    }
226 227 228

    connect(&windowNameUpdateTimer, SIGNAL(timeout()), this, SLOT(configureWindowName()));
    windowNameUpdateTimer.start(15000);
lm's avatar
lm committed
229 230
    emit initStatusChanged("Done.");
    show();
pixhawk's avatar
pixhawk committed
231 232
}

pixhawk's avatar
pixhawk committed
233
MainWindow::~MainWindow()
pixhawk's avatar
pixhawk committed
234
{
235
    delete mavlink;
236
    delete joystick;
lm's avatar
lm committed
237

238 239 240 241 242 243
    // Get and delete all dockwidgets and contained
    // widgets
    QObjectList childList( this->children() );

    QObjectList::iterator i;
    QDockWidget* dockWidget;
244 245
    for (i = childList.begin(); i != childList.end(); ++i)
    {
246
        dockWidget = dynamic_cast<QDockWidget*>(*i);
247 248
        if (dockWidget)
        {
249
            // Remove dock widget from main window
250 251
            // removeDockWidget(dockWidget);
            // delete dockWidget->widget();
252 253
            delete dockWidget;
        }
LM's avatar
LM committed
254 255 256 257
        else
        {
            delete dynamic_cast<QObject*>(*i);
        }
258 259
    }

LM's avatar
LM committed
260
    // Delete all UAS objects
261 262
}

lm's avatar
lm committed
263 264
void MainWindow::resizeEvent(QResizeEvent * event)
{
265 266
    if (height() < 800)
    {
lm's avatar
lm committed
267
        ui.statusBar->setVisible(false);
268 269 270
    }
    else
    {
lm's avatar
lm committed
271
        ui.statusBar->setVisible(true);
272
        ui.statusBar->setSizeGripEnabled(true);
lm's avatar
lm committed
273
    }
274 275 276 277 278 279 280 281 282 283 284

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

    QMainWindow::resizeEvent(event);
lm's avatar
lm committed
285 286
}

287 288
QString MainWindow::getWindowStateKey()
{
289
    return QString::number(currentView)+"_windowstate";
290 291 292 293
}

QString MainWindow::getWindowGeometryKey()
{
294 295
    //return QString::number(currentView)+"_geometry";
    return "_geometry";
296 297
}

lm's avatar
lm committed
298 299
void MainWindow::buildCustomWidget()
{
lm's avatar
lm committed
300 301
    // Create custom widgets
    QList<QGCToolWidget*> widgets = QGCToolWidget::createWidgetsFromSettings(this);
lm's avatar
lm committed
302

lm's avatar
lm committed
303 304 305 306
    if (widgets.size() > 0)
    {
        ui.menuTools->addSeparator();
    }
lm's avatar
lm committed
307

lm's avatar
lm committed
308 309 310 311 312 313
    for(int i = 0; i < widgets.size(); ++i)
    {
        // Check if this widget already has a parent, do not create it in this case
        QGCToolWidget* tool = widgets.at(i);
        QDockWidget* dock = dynamic_cast<QDockWidget*>(tool->parentWidget());
        if (!dock)
314
        {
lm's avatar
lm committed
315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
            QDockWidget* dock = new QDockWidget(tool->windowTitle(), this);
            dock->setObjectName(tool->objectName()+"_DOCK");
            dock->setWidget(tool);
            connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
            QAction* showAction = new QAction(widgets.at(i)->windowTitle(), this);
            showAction->setCheckable(true);
            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);

            // 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
330 331 332 333
        }
    }
}

334 335 336 337
void MainWindow::buildCommonWidgets()
{
    //TODO:  move protocol outside UI
    mavlink     = new MAVLinkProtocol();
338
    connect(mavlink, SIGNAL(protocolStatusMessage(QString,QString)), this, SLOT(showCriticalMessage(QString,QString)), Qt::QueuedConnection);
LM's avatar
LM committed
339 340
    // Add generic MAVLink decoder
    mavlinkDecoder = new MAVLinkDecoder(mavlink, this);
341 342

    // Dock widgets
343 344
    if (!controlDockWidget)
    {
345
        controlDockWidget = new QDockWidget(tr("Control"), this);
346
        controlDockWidget->setObjectName("UNMANNED_SYSTEM_CONTROL_DOCKWIDGET");
347
        controlDockWidget->setWidget( new UASControlWidget(this) );
348
        addTool(controlDockWidget, tr("Control"), Qt::LeftDockWidgetArea);
349
    }
350

351 352
    if (!listDockWidget)
    {
353 354
        listDockWidget = new QDockWidget(tr("Unmanned Systems"), this);
        listDockWidget->setWidget( new UASListWidget(this) );
355
        listDockWidget->setObjectName("UNMANNED_SYSTEMS_LIST_DOCKWIDGET");
356
        addTool(listDockWidget, tr("Unmanned Systems"), Qt::RightDockWidgetArea);
357
    }
358

359 360
    if (!waypointsDockWidget)
    {
361
        waypointsDockWidget = new QDockWidget(tr("Mission Plan"), this);
362
        waypointsDockWidget->setWidget( new QGCWaypointListMulti(this) );
363
        waypointsDockWidget->setObjectName("WAYPOINT_LIST_DOCKWIDGET");
364
        addTool(waypointsDockWidget, tr("Mission Plan"), Qt::BottomDockWidgetArea);
365
    }
366

367 368
    if (!infoDockWidget)
    {
369 370
        infoDockWidget = new QDockWidget(tr("Status Details"), this);
        infoDockWidget->setWidget( new UASInfoWidget(this) );
pixhawk's avatar
pixhawk committed
371
        infoDockWidget->setObjectName("UAS_STATUS_DETAILS_DOCKWIDGET");
372
        addTool(infoDockWidget, tr("Status Details"), Qt::RightDockWidgetArea);
373
    }
374

375 376
    if (!debugConsoleDockWidget)
    {
377 378
        debugConsoleDockWidget = new QDockWidget(tr("Communication Console"), this);
        debugConsoleDockWidget->setWidget( new DebugConsole(this) );
379
        debugConsoleDockWidget->setObjectName("COMMUNICATION_DEBUG_CONSOLE_DOCKWIDGET");
LM's avatar
LM committed
380 381 382 383

        DebugConsole *debugConsole = dynamic_cast<DebugConsole*>(debugConsoleDockWidget->widget());
        connect(mavlinkDecoder, SIGNAL(textMessageReceived(int, int, int, const QString)), debugConsole, SLOT(receiveTextMessage(int, int, int, const QString)));

384
        addTool(debugConsoleDockWidget, tr("Communication Console"), Qt::BottomDockWidgetArea);
385
    }
386

387 388
    if (!logPlayerDockWidget)
    {
389
        logPlayerDockWidget = new QDockWidget(tr("MAVLink Log Player"), this);
lm's avatar
lm committed
390 391 392
        logPlayer = new QGCMAVLinkLogPlayer(mavlink, this);
        toolBar->setLogPlayer(logPlayer);
        logPlayerDockWidget->setWidget(logPlayer);
393
        logPlayerDockWidget->setObjectName("MAVLINK_LOG_PLAYER_DOCKWIDGET");
394
        addTool(logPlayerDockWidget, tr("MAVLink Log Replay"), Qt::RightDockWidgetArea);
395 396
    }

lm's avatar
lm committed
397 398 399 400 401
    if (!mavlinkInspectorWidget)
    {
        mavlinkInspectorWidget = new QDockWidget(tr("MAVLink Message Inspector"), this);
        mavlinkInspectorWidget->setWidget( new QGCMAVLinkInspector(mavlink, this) );
        mavlinkInspectorWidget->setObjectName("MAVLINK_INSPECTOR_DOCKWIDGET");
lm's avatar
lm committed
402
        addTool(mavlinkInspectorWidget, tr("MAVLink Inspector"), Qt::RightDockWidgetArea);
lm's avatar
lm committed
403 404
    }

pixhawk's avatar
pixhawk committed
405 406
    //FIXME: memory of acceptList will never be freed again
    QStringList* acceptList = new QStringList();
pixhawk's avatar
pixhawk committed
407 408 409
    acceptList->append("-105,roll deg,deg,+105,s");
    acceptList->append("-105,pitch deg,deg,+105,s");
    acceptList->append("-105,heading deg,deg,+105,s");
410

pixhawk's avatar
pixhawk committed
411 412
    //FIXME: memory of acceptList2 will never be freed again
    QStringList* acceptList2 = new QStringList();
lm's avatar
lm committed
413
    acceptList2->append("0,abs pressure,hPa,65500");
414

lm's avatar
lm committed
415 416 417 418 419 420
    if (!parametersDockWidget) {
        parametersDockWidget = new QDockWidget(tr("Calibration and Onboard Parameters"), this);
        parametersDockWidget->setWidget( new ParameterInterface(this) );
        parametersDockWidget->setObjectName("PARAMETER_INTERFACE_DOCKWIDGET");
        addTool(parametersDockWidget, tr("Calibration and Parameters"), Qt::RightDockWidgetArea);
    }
421
	
lm's avatar
lm committed
422 423 424 425 426 427
    if (!hsiDockWidget) {
        hsiDockWidget = new QDockWidget(tr("Horizontal Situation Indicator"), this);
        hsiDockWidget->setWidget( new HSIDisplay(this) );
        hsiDockWidget->setObjectName("HORIZONTAL_SITUATION_INDICATOR_DOCK_WIDGET");
        addTool(hsiDockWidget, tr("Horizontal Situation"), Qt::BottomDockWidgetArea);
    }
428
	
lm's avatar
lm committed
429 430
    if (!headDown1DockWidget) {
        headDown1DockWidget = new QDockWidget(tr("Flight Display"), this);
431 432 433
        HDDisplay* hdDisplay = new HDDisplay(acceptList, "Flight Display", this);
        hdDisplay->addSource(mavlinkDecoder);
        headDown1DockWidget->setWidget(hdDisplay);
lm's avatar
lm committed
434 435 436 437 438 439
        headDown1DockWidget->setObjectName("HEAD_DOWN_DISPLAY_1_DOCK_WIDGET");
        addTool(headDown1DockWidget, tr("Flight Display"), Qt::RightDockWidgetArea);
    }

    if (!headDown2DockWidget) {
        headDown2DockWidget = new QDockWidget(tr("Actuator Status"), this);
440 441 442
        HDDisplay* hdDisplay = new HDDisplay(acceptList2, "Actuator Status", this);
        hdDisplay->addSource(mavlinkDecoder);
        headDown2DockWidget->setWidget(hdDisplay);
lm's avatar
lm committed
443 444 445
        headDown2DockWidget->setObjectName("HEAD_DOWN_DISPLAY_2_DOCK_WIDGET");
        addTool(headDown2DockWidget, tr("Actuator Status"), Qt::RightDockWidgetArea);
    }
446
	
lm's avatar
lm committed
447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462
    if (!rcViewDockWidget) {
        rcViewDockWidget = new QDockWidget(tr("Radio Control"), this);
        rcViewDockWidget->setWidget( new QGCRemoteControlView(this) );
        rcViewDockWidget->setObjectName("RADIO_CONTROL_CHANNELS_DOCK_WIDGET");
        addTool(rcViewDockWidget, tr("Radio Control"), Qt::BottomDockWidgetArea);
    }

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

    if (!video1DockWidget) {
        video1DockWidget = new QDockWidget(tr("Video Stream 1"), this);
463
        QGCRGBDView* video1 =  new QGCRGBDView(160, 120, this);
lm's avatar
lm committed
464
        video1->enableHUDInstruments(false);
465
        video1->enableVideo(false);
lm's avatar
lm committed
466 467 468 469 470 471 472 473
        // FIXME select video stream as well
        video1DockWidget->setWidget(video1);
        video1DockWidget->setObjectName("VIDEO_STREAM_1_DOCK_WIDGET");
        addTool(video1DockWidget, tr("Video Stream 1"), Qt::LeftDockWidgetArea);
    }

    if (!video2DockWidget) {
        video2DockWidget = new QDockWidget(tr("Video Stream 2"), this);
474
        QGCRGBDView* video2 =  new QGCRGBDView(160, 120, this);
lm's avatar
lm committed
475
        video2->enableHUDInstruments(false);
476
        video2->enableVideo(false);
lm's avatar
lm committed
477 478 479 480 481 482
        // FIXME select video stream as well
        video2DockWidget->setWidget(video2);
        video2DockWidget->setObjectName("VIDEO_STREAM_2_DOCK_WIDGET");
        addTool(video2DockWidget, tr("Video Stream 2"), Qt::LeftDockWidgetArea);
    }

483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504
//    if (!rgbd1DockWidget) {
//        rgbd1DockWidget = 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");
//        addTool(video1DockWidget, tr("Video Stream 1"), Qt::LeftDockWidgetArea);
//    }

//    if (!rgbd2DockWidget) {
//        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");
//        addTool(video2DockWidget, tr("Video Stream 2"), Qt::LeftDockWidgetArea);
//    }

lm's avatar
lm committed
505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520
    // Custom widgets, added last to all menus and layouts
    buildCustomWidget();

    // Center widgets
    if (!mapWidget)
    {
        mapWidget = new QGCMapTool(this);
        addCentralWidget(mapWidget, "Maps");
    }

    if (!protocolWidget)
    {
        protocolWidget    = new XMLCommProtocolWidget(this);
        addCentralWidget(protocolWidget, "Mavlink Generator");
    }

521
    if (!hudWidget) {
522
        hudWidget         = new HUD(320, 240, this);
523
        addCentralWidget(hudWidget, tr("Head Up Display"));
lm's avatar
lm committed
524 525
    }

526
    if (!dataplotWidget) {
527
        dataplotWidget    = new QGCDataPlot2D(this);
528
        addCentralWidget(dataplotWidget, tr("Logfile Plot"));
529
    }
530

531
#ifdef QGC_OSG_ENABLED
532
    if (!_3DWidget) {
533
        _3DWidget         = Q3DWidgetFactory::get("PIXHAWK");
534
        addCentralWidget(_3DWidget, tr("Local 3D"));
535
    }
536
#endif
537

538
#if (defined _MSC_VER) | (defined Q_OS_MAC)
539
    if (!gEarthWidget) {
540
        gEarthWidget = new QGCGoogleEarthView(this);
541
        addCentralWidget(gEarthWidget, tr("Google Earth"));
542
    }
543
#endif
lm's avatar
lm committed
544
}
545

546
void MainWindow::addTool(QDockWidget* widget, const QString& title, Qt::DockWidgetArea area)
lm's avatar
lm committed
547
{
548
    QAction* tempAction = ui.menuTools->addAction(title);
549

550
    tempAction->setCheckable(true);
551 552 553 554
    QVariant var;
    var.setValue((QWidget*)widget);
    tempAction->setData(var);
    connect(tempAction,SIGNAL(triggered(bool)),this, SLOT(showTool(bool)));
LM's avatar
LM committed
555 556
    connect(widget, SIGNAL(visibilityChanged(bool)), tempAction, SLOT(setChecked(bool)));
    tempAction->setChecked(widget->isVisible());
557
    addDockWidget(area, widget);
558 559 560
}


561
void MainWindow::showTool(bool show)
lm's avatar
lm committed
562
{
563 564 565
    QAction* act = qobject_cast<QAction *>(sender());
    QWidget* widget = qVariantValue<QWidget *>(act->data());
    widget->setVisible(show);
566
}
567

568
void MainWindow::addCentralWidget(QWidget* widget, const QString& title)
lm's avatar
lm committed
569
{
570 571 572 573
    // Check if this widget already has been added
    if (centerStack->indexOf(widget) == -1)
    {
        centerStack->addWidget(widget);
574

575
        QAction* tempAction = ui.menuMain->addAction(title);
576

577 578 579 580
        tempAction->setCheckable(true);
        QVariant var;
        var.setValue((QWidget*)widget);
        tempAction->setData(var);
581
        centerStackActionGroup->addAction(tempAction);
582
        connect(tempAction,SIGNAL(triggered()),this, SLOT(showCentralWidget()));
LM's avatar
LM committed
583 584
        connect(widget, SIGNAL(visibilityChanged(bool)), tempAction, SLOT(setChecked(bool)));
        tempAction->setChecked(widget->isVisible());
585 586 587 588
    }
}


589
void MainWindow::showCentralWidget()
590
{
591 592 593
    QAction* act = qobject_cast<QAction *>(sender());
    QWidget* widget = qVariantValue<QWidget *>(act->data());
    centerStack->setCurrentWidget(widget);
594 595
}

596 597
void MainWindow::closeEvent(QCloseEvent *event)
{
pixhawk's avatar
pixhawk committed
598
    if (isVisible()) storeViewState();
599
    storeSettings();
600
    aboutToCloseFlag = true;
601
    mavlink->storeSettings();
602
    UASManager::instance()->storeSettings();
603 604
    QMainWindow::closeEvent(event);
}
605 606 607 608

/**
 * Connect the signals and slots of the common window widgets
 */
609
void MainWindow::connectCommonWidgets()
610
{
611
    if (infoDockWidget && infoDockWidget->widget()) {
pixhawk's avatar
pixhawk committed
612 613 614
        connect(mavlink, SIGNAL(receiveLossChanged(int, float)),
                infoDockWidget->widget(), SLOT(updateSendLoss(int, float)));
    }
615
}
616

617 618
void MainWindow::createCustomWidget()
{
619 620
    QDockWidget* dock = new QDockWidget("Unnamed Tool", this);
    QGCToolWidget* tool = new QGCToolWidget("Unnamed Tool", dock);
lm's avatar
lm committed
621

LM's avatar
LM committed
622 623
    if (QGCToolWidget::instances()->size() < 2)
    {
lm's avatar
lm committed
624 625 626 627 628
        // This is the first widget
        ui.menuTools->addSeparator();
    }

    connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
629
    dock->setWidget(tool);
630

631 632 633 634 635 636 637 638 639 640 641 642 643 644
    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));
645
    if (fileName != "") loadCustomWidget(fileName);
LM's avatar
LM committed
646
}
647

LM's avatar
LM committed
648 649 650 651
void MainWindow::loadCustomWidget(const QString& fileName, bool singleinstance)
{
    QGCToolWidget* tool = new QGCToolWidget("", this);
    if (tool->loadSettings(fileName, true) || !singleinstance)
LM's avatar
LM committed
652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667
    {
        // Add widget to UI
        QDockWidget* dock = new QDockWidget(tool->getTitle(), this);
        connect(tool, SIGNAL(destroyed()), dock, SLOT(deleteLater()));
        dock->setWidget(tool);
        tool->setParent(dock);

        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);
    }
LM's avatar
LM committed
668 669 670
    else
    {
        return;
671
    }
672 673
}

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

LM's avatar
LM committed
678 679 680 681 682 683
    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;
lm's avatar
lm committed
684
        showStatusMessage(tr("Did not find any custom widgets in %1").arg(defaultsDir));
685
    }
686

LM's avatar
LM committed
687 688
    // Load all custom widgets found in the AP folder
    for(int i = 0; i < files.count(); ++i)
lm's avatar
lm committed
689
    {
LM's avatar
LM committed
690 691 692 693 694
        QString file = files[i];
        if (file.endsWith(".qgw"))
        {
            // Will only be loaded if not already a custom widget with
            // the same name is present
695
            loadCustomWidget(defaultsDir+"/"+file, true);
lm's avatar
lm committed
696
            showStatusMessage(tr("Loaded custom widget %1").arg(defaultsDir+"/"+file));
LM's avatar
LM committed
697
        }
lm's avatar
lm committed
698
    }
699 700
}

701 702 703 704 705 706
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();
707
    lowPowerMode = settings.value("LOW_POWER_MODE", lowPowerMode).toBool();
708 709 710 711 712 713 714 715 716 717
    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
718 719 720 721 722 723 724 725 726 727 728
    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
    }
729
    settings.setValue("LOW_POWER_MODE", lowPowerMode);
730 731 732
    settings.sync();
}

733 734
void MainWindow::configureWindowName()
{
pixhawk's avatar
pixhawk committed
735 736 737
    QList<QHostAddress> hostAddresses = QNetworkInterface::allAddresses();
    QString windowname = qApp->applicationName() + " " + qApp->applicationVersion();
    bool prevAddr = false;
738

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

741 742
    for (int i = 0; i < hostAddresses.size(); i++)
    {
pixhawk's avatar
pixhawk committed
743
        // Exclude loopback IPv4 and all IPv6 addresses
744 745
        if (hostAddresses.at(i) != QHostAddress("127.0.0.1") && !hostAddresses.at(i).toString().contains(":"))
        {
pixhawk's avatar
pixhawk committed
746 747 748 749 750
            if(prevAddr) windowname.append("/");
            windowname.append(hostAddresses.at(i).toString());
            prevAddr = true;
        }
    }
751

pixhawk's avatar
pixhawk committed
752
    windowname.append(")");
753

pixhawk's avatar
pixhawk committed
754
    setWindowTitle(windowname);
755 756

#ifndef Q_WS_MAC
pixhawk's avatar
pixhawk committed
757
    //qApp->setWindowIcon(QIcon(":/core/images/qtcreator_logo_128.png"));
758 759 760
#endif
}

pixhawk's avatar
pixhawk committed
761
void MainWindow::startVideoCapture()
pixhawk's avatar
pixhawk committed
762 763 764 765 766
{
    QString format = "bmp";
    QString initialPath = QDir::currentPath() + tr("/untitled.") + format;

    QString screenFileName = QFileDialog::getSaveFileName(this, tr("Save As"),
lm's avatar
lm committed
767 768 769 770
                                                          initialPath,
                                                          tr("%1 Files (*.%2);;All Files (*)")
                                                          .arg(format.toUpper())
                                                          .arg(format));
pixhawk's avatar
pixhawk committed
771 772
    delete videoTimer;
    videoTimer = new QTimer(this);
773 774 775
    //videoTimer->setInterval(40);
    //connect(videoTimer, SIGNAL(timeout()), this, SLOT(saveScreen()));
    //videoTimer->stop();
pixhawk's avatar
pixhawk committed
776 777
}

pixhawk's avatar
pixhawk committed
778
void MainWindow::stopVideoCapture()
pixhawk's avatar
pixhawk committed
779 780 781 782 783 784
{
    videoTimer->stop();

    // TODO Convert raw images to PNG
}

pixhawk's avatar
pixhawk committed
785
void MainWindow::saveScreen()
pixhawk's avatar
pixhawk committed
786 787 788 789
{
    QPixmap window = QPixmap::grabWindow(this->winId());
    QString format = "bmp";

790 791
    if (!screenFileName.isEmpty())
    {
pixhawk's avatar
pixhawk committed
792 793 794 795
        window.save(screenFileName, format.toAscii());
    }
}

796
void MainWindow::enableAutoReconnect(bool enabled)
pixhawk's avatar
pixhawk committed
797
{
798 799
    autoReconnect = enabled;
}
800

801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817
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)
{
818 819 820 821 822 823 824 825 826
    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."));
827
        }
828 829
    }
    break;
830 831 832 833 834 835 836 837 838 839 840 841 842 843
    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;
}

844
void MainWindow::selectStylesheet()
pixhawk's avatar
pixhawk committed
845
{
846
    // Let user select style sheet
847
    styleFileName = QFileDialog::getOpenFileName(this, tr("Specify stylesheet"), styleFileName, tr("CSS Stylesheet (*.css);;"));
848

849
    if (!styleFileName.endsWith(".css")) {
850 851 852 853 854 855 856 857 858 859
        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
860
    // Load style sheet
861 862 863 864 865 866 867
    reloadStylesheet();
}

void MainWindow::reloadStylesheet()
{
    // Load style sheet
    QFile* styleSheet = new QFile(styleFileName);
868
    if (!styleSheet->exists()) {
869 870
        styleSheet = new QFile(":/images/style-mission.css");
    }
871
    if (styleSheet->open(QIODevice::ReadOnly | QIODevice::Text)) {
872 873
        QString style = QString(styleSheet->readAll());
        style.replace("ICONDIR", QCoreApplication::applicationDirPath()+ "/images/");
pixhawk's avatar
pixhawk committed
874
        qApp->setStyleSheet(style);
875
    } else {
876 877 878
        QMessageBox msgBox;
        msgBox.setIcon(QMessageBox::Information);
        msgBox.setText(tr("QGroundControl did lot load a new style"));
879
        msgBox.setInformativeText(tr("Stylesheet file %1 was not readable").arg(styleFileName));
880 881 882
        msgBox.setStandardButtons(QMessageBox::Ok);
        msgBox.setDefaultButton(QMessageBox::Ok);
        msgBox.exec();
pixhawk's avatar
pixhawk committed
883
    }
884
    delete styleSheet;
pixhawk's avatar
pixhawk committed
885 886
}

887 888 889 890 891 892
/**
 * 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
893
void MainWindow::showStatusMessage(const QString& status, int timeout)
pixhawk's avatar
pixhawk committed
894
{
lm's avatar
lm committed
895
    statusBar()->showMessage(status, timeout);
pixhawk's avatar
pixhawk committed
896 897
}

898 899 900 901 902 903
/**
 * 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
 */
904
void MainWindow::showStatusMessage(const QString& status)
pixhawk's avatar
pixhawk committed
905
{
lm's avatar
lm committed
906
    statusBar()->showMessage(status, 20000);
907 908 909 910
}

void MainWindow::showCriticalMessage(const QString& title, const QString& message)
{
911
    QMessageBox msgBox(this);
912 913 914 915 916 917
    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
918 919
}

lm's avatar
lm committed
920 921
void MainWindow::showInfoMessage(const QString& title, const QString& message)
{
922
    QMessageBox msgBox(this);
lm's avatar
lm committed
923 924 925 926 927 928 929 930
    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
931 932 933 934
/**
* @brief Create all actions associated to the main window
*
**/
935
void MainWindow::connectCommonActions()
pixhawk's avatar
pixhawk committed
936
{
937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959
    // 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
960 961 962 963 964
    // 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*)));
965
    connect(UASManager::instance(), SIGNAL(activeUASSet(UASInterface*)), this, SLOT(setActiveUAS(UASInterface*)));
pixhawk's avatar
pixhawk committed
966

967
    // Unmanned System controls
pixhawk's avatar
pixhawk committed
968 969 970 971
    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()));
972
    connect(ui.actionShutdownMAV, SIGNAL(triggered()), UASManager::instance(), SLOT(shutdownActiveUAS()));
pixhawk's avatar
pixhawk committed
973 974
    connect(ui.actionConfiguration, SIGNAL(triggered()), UASManager::instance(), SLOT(configureActiveUAS()));

975
    // Views actions
976
    connect(ui.actionPilotsView, SIGNAL(triggered()), this, SLOT(loadPilotView()));
977
    connect(ui.actionEngineersView, SIGNAL(triggered()), this, SLOT(loadEngineerView()));
978
    connect(ui.actionOperatorsView, SIGNAL(triggered()), this, SLOT(loadOperatorView()));
979
    connect(ui.actionUnconnectedView, SIGNAL(triggered()), this, SLOT(loadUnconnectedView()));
980

981
    connect(ui.actionMavlinkView, SIGNAL(triggered()), this, SLOT(loadMAVLinkView()));
982 983
    connect(ui.actionReloadStylesheet, SIGNAL(triggered()), this, SLOT(reloadStylesheet()));
    connect(ui.actionSelectStylesheet, SIGNAL(triggered()), this, SLOT(selectStylesheet()));
984

985
    // Help Actions
986
    connect(ui.actionOnline_Documentation, SIGNAL(triggered()), this, SLOT(showHelp()));
987
    connect(ui.actionDeveloper_Credits, SIGNAL(triggered()), this, SLOT(showCredits()));
988
    connect(ui.actionProject_Roadmap_2, SIGNAL(triggered()), this, SLOT(showRoadMap()));
989 990 991

    // Custom widget actions
    connect(ui.actionNewCustomWidget, SIGNAL(triggered()), this, SLOT(createCustomWidget()));
992
    connect(ui.actionLoadCustomWidgetFile, SIGNAL(triggered()), this, SLOT(loadCustomWidget()));
993 994 995

    // Audio output
    ui.actionMuteAudioOutput->setChecked(GAudioOutput::instance()->isMuted());
996
    connect(GAudioOutput::instance(), SIGNAL(mutedChanged(bool)), ui.actionMuteAudioOutput, SLOT(setChecked(bool)));
997
    connect(ui.actionMuteAudioOutput, SIGNAL(triggered(bool)), GAudioOutput::instance(), SLOT(mute(bool)));
998 999 1000 1001 1002 1003 1004

    // 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);
1005 1006 1007

    // Configuration
    // Joystick
1008
    connect(ui.actionJoystickSettings, SIGNAL(triggered()), this, SLOT(configure()));
1009 1010
    // Application Settings
    connect(ui.actionSettings, SIGNAL(triggered()), this, SLOT(showSettings()));
1011 1012
}

1013 1014
void MainWindow::showHelp()
{
lm's avatar
lm committed
1015
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/users/start"))) {
1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027
        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()
{
1028
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/credits"))) {
1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040
        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()
{
lm's avatar
lm committed
1041
    if(!QDesktopServices::openUrl(QUrl("http://qgroundcontrol.org/dev/roadmap"))) {
1042 1043 1044 1045 1046 1047 1048 1049 1050 1051
        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
1052
void MainWindow::configure()
pixhawk's avatar
pixhawk committed
1053
{
1054 1055
    if (!joystickWidget) {
        if (!joystick->isRunning()) {
1056 1057 1058 1059 1060
            joystick->start();
        }
        joystickWidget = new JoystickWidget(joystick);
    }
    joystickWidget->show();
pixhawk's avatar
pixhawk committed
1061 1062
}

1063 1064 1065 1066 1067 1068
void MainWindow::showSettings()
{
    QGCSettingsWidget* settings = new QGCSettingsWidget(this);
    settings->show();
}

pixhawk's avatar
pixhawk committed
1069
void MainWindow::addLink()
pixhawk's avatar
pixhawk committed
1070 1071 1072 1073
{
    SerialLink* link = new SerialLink();
    // TODO This should be only done in the dialog itself

1074
    LinkManager::instance()->add(link);
pixhawk's avatar
pixhawk committed
1075 1076
    LinkManager::instance()->addProtocol(link, mavlink);

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

oberion's avatar
oberion committed
1080 1081
	const int32_t& linkIndex(LinkManager::instance()->getLinks().indexOf(link));
	const int32_t& linkID(LinkManager::instance()->getLinks()[linkIndex]->getId());
oberion's avatar
oberion committed
1082

1083
    foreach (QAction* act, actions) {
oberion's avatar
oberion committed
1084
        if (act->data().toInt() == linkID) { // LinkManager::instance()->getLinks().indexOf(link)
1085 1086 1087 1088
            act->trigger();
            break;
        }
    }
pixhawk's avatar
pixhawk committed
1089 1090
}

pixhawk's avatar
pixhawk committed
1091 1092
void MainWindow::addLink(LinkInterface *link)
{
1093 1094 1095 1096
    // 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)
1097
    LinkManager::instance()->add(link);
1098
    LinkManager::instance()->addProtocol(link, mavlink);
1099

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

oberion's avatar
oberion committed
1103
    bool found(false);
1104

oberion's avatar
oberion committed
1105 1106
	const int32_t& linkIndex(LinkManager::instance()->getLinks().indexOf(link));
	const int32_t& linkID(LinkManager::instance()->getLinks()[linkIndex]->getId());
1107

1108
    foreach (QAction* act, actions) {
oberion's avatar
oberion committed
1109
        if (act->data().toInt() == linkID) { // LinkManager::instance()->getLinks().indexOf(link)
1110 1111 1112 1113
            found = true;
        }
    }

oberion's avatar
oberion committed
1114
    //UDPLink* udp = dynamic_cast<UDPLink*>(link);
1115

oberion's avatar
oberion committed
1116
    if (!found) {  //  || udp
1117 1118 1119 1120 1121 1122 1123 1124
        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);
1125 1126
        if (sim)
        {
1127 1128 1129
            //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
1130 1131 1132
    }
}

1133 1134 1135 1136 1137 1138 1139
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);
}

1140 1141 1142
void MainWindow::UASSpecsChanged(int uas)
{
    UASInterface* activeUAS = UASManager::instance()->getActiveUAS();
1143 1144 1145 1146
    if (activeUAS)
    {
        if (activeUAS->getUASID() == uas)
        {
1147 1148 1149 1150 1151
            ui.menuUnmanned_System->setTitle(activeUAS->getUASName());
        }
    }
}

pixhawk's avatar
pixhawk committed
1152
void MainWindow::UASCreated(UASInterface* uas)
pixhawk's avatar
pixhawk committed
1153
{
1154

pixhawk's avatar
pixhawk committed
1155 1156
    // Connect the UAS to the full user interface

lm's avatar
lm committed
1157 1158
    if (uas != NULL)
    {
1159
        // The pilot, operator and engineer views were not available on startup, enable them now
1160
        ui.actionPilotsView->setEnabled(true);
1161 1162
        ui.actionOperatorsView->setEnabled(true);
        ui.actionEngineersView->setEnabled(true);
1163 1164 1165 1166 1167 1168
        // 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);
1169

1170 1171
        QIcon icon;
        // Set matching icon
1172
        switch (uas->getSystemType()) {
1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195
        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;
        }

1196 1197 1198
        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()));
1199
        connect(uas, SIGNAL(systemSpecsChanged(int)), this, SLOT(UASSpecsChanged(int)));
1200 1201

        ui.menuConnected_Systems->addAction(uasAction);
1202 1203

        // FIXME Should be not inside the mainwindow
1204 1205
        if (debugConsoleDockWidget)
        {
pixhawk's avatar
pixhawk committed
1206
            DebugConsole *debugConsole = dynamic_cast<DebugConsole*>(debugConsoleDockWidget->widget());
1207 1208
            if (debugConsole)
            {
pixhawk's avatar
pixhawk committed
1209 1210 1211
                connect(uas, SIGNAL(textMessageReceived(int,int,int,QString)),
                        debugConsole, SLOT(receiveTextMessage(int,int,int,QString)));
            }
1212
        }
1213 1214

        // Health / System status indicator
1215 1216
        if (infoDockWidget)
        {
pixhawk's avatar
pixhawk committed
1217
            UASInfoWidget *infoWidget = dynamic_cast<UASInfoWidget*>(infoDockWidget->widget());
1218 1219
            if (infoWidget)
            {
pixhawk's avatar
pixhawk committed
1220 1221
                infoWidget->addUAS(uas);
            }
1222
        }
1223 1224

        // UAS List
lm's avatar
lm committed
1225 1226
        if (listDockWidget)
        {
pixhawk's avatar
pixhawk committed
1227
            UASListWidget *listWidget = dynamic_cast<UASListWidget*>(listDockWidget->widget());
lm's avatar
lm committed
1228 1229
            if (listWidget)
            {
pixhawk's avatar
pixhawk committed
1230 1231
                listWidget->addUAS(uas);
            }
1232
        }
1233

1234 1235
        // Line chart
        if (!linechartWidget)
lm's avatar
lm committed
1236
        {
1237
            // Center widgets
LM's avatar
LM committed
1238
            linechartWidget = new Linecharts(this);
1239
            linechartWidget->addSource(mavlinkDecoder);
1240
            addCentralWidget(linechartWidget, tr("Realtime Plot"));
1241
        }
1242

LM's avatar
LM committed
1243
        // Load default custom widgets for this autopilot type
lm's avatar
lm committed
1244
        loadCustomWidgetsFromDefaults(uas->getSystemTypeName(), uas->getAutopilotTypeName());
1245 1246


lm's avatar
lm committed
1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257
        if (uas->getAutopilotType() == MAV_AUTOPILOT_PIXHAWK)
        {
            // Dock widgets
            if (!detectionDockWidget)
            {
                detectionDockWidget = new QDockWidget(tr("Object Recognition"), this);
                detectionDockWidget->setWidget( new ObjectDetectionView("images/patterns", this) );
                detectionDockWidget->setObjectName("OBJECT_DETECTION_DOCK_WIDGET");
                addTool(detectionDockWidget, tr("Object Recognition"), Qt::RightDockWidgetArea);
            }

1258 1259
            if (!watchdogControlDockWidget)
            {
lm's avatar
lm committed
1260 1261 1262 1263 1264
                watchdogControlDockWidget = new QDockWidget(tr("Process Control"), this);
                watchdogControlDockWidget->setWidget( new WatchdogControl(this) );
                watchdogControlDockWidget->setObjectName("WATCHDOG_CONTROL_DOCKWIDGET");
                addTool(watchdogControlDockWidget, tr("Process Control"), Qt::BottomDockWidgetArea);
            }
1265 1266 1267 1268 1269 1270
        }

        // 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
1271 1272
        if (UASManager::instance()->getUASList().size() == 1)
        {
1273
            // Load last view if setting is present
1274 1275
            if (settings.contains("CURRENT_VIEW_WITH_UAS_CONNECTED"))
            {
1276
                int view = settings.value("CURRENT_VIEW_WITH_UAS_CONNECTED").toInt();
1277
                switch (view) {
1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293
                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;
1294
                }
1295 1296 1297
            }
            else
            {
1298
                loadOperatorView();
1299
            }
1300
        }
1301

1302
    }
1303 1304

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

1306 1307
    // Reload view state in case new widgets were added
    loadViewState();
pixhawk's avatar
pixhawk committed
1308 1309
}

1310
/**
1311
 * Stores the current view state
1312
 */
1313
void MainWindow::storeViewState()
1314
{
pixhawk's avatar
pixhawk committed
1315
    if (!aboutToCloseFlag)
lm's avatar
lm committed
1316
    {
pixhawk's avatar
pixhawk committed
1317 1318 1319 1320 1321 1322 1323 1324
        // 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());
1325
    }
1326
}
1327

1328 1329 1330
void MainWindow::loadViewState()
{
    // Restore center stack state
lm's avatar
lm committed
1331
    int index = settings.value(getWindowStateKey()+"CENTER_WIDGET", -1).toInt();
LM's avatar
LM committed
1332 1333 1334 1335 1336 1337
    // 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);
    }
pixhawk's avatar
pixhawk committed
1338

lm's avatar
lm committed
1339 1340 1341 1342 1343 1344
    if (index != -1)
    {
        centerStack->setCurrentIndex(index);
    }
    else
    {
1345 1346 1347 1348
        // Hide custom widgets
        if (detectionDockWidget) detectionDockWidget->hide();
        if (watchdogControlDockWidget) watchdogControlDockWidget->hide();

lm's avatar
lm committed
1349 1350 1351 1352 1353
        // Load defaults
        switch (currentView)
        {
        case VIEW_ENGINEER:
            centerStack->setCurrentWidget(linechartWidget);
lm's avatar
lm committed
1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368
            controlDockWidget->hide();
            listDockWidget->hide();
            waypointsDockWidget->hide();
            infoDockWidget->hide();
            debugConsoleDockWidget->show();
            logPlayerDockWidget->show();
            mavlinkInspectorWidget->show();
            parametersDockWidget->show();
            hsiDockWidget->hide();
            headDown1DockWidget->hide();
            headDown2DockWidget->hide();
            rcViewDockWidget->hide();
            headUpDockWidget->hide();
            video1DockWidget->hide();
            video2DockWidget->hide();
lm's avatar
lm committed
1369 1370 1371
            break;
        case VIEW_PILOT:
            centerStack->setCurrentWidget(hudWidget);
lm's avatar
lm committed
1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384
            controlDockWidget->hide();
            listDockWidget->hide();
            waypointsDockWidget->hide();
            infoDockWidget->hide();
            debugConsoleDockWidget->hide();
            logPlayerDockWidget->hide();
            mavlinkInspectorWidget->hide();
            parametersDockWidget->hide();
            hsiDockWidget->show();
            headDown1DockWidget->show();
            headDown2DockWidget->show();
            rcViewDockWidget->hide();
            headUpDockWidget->hide();
1385
            video1DockWidget->hide();
lm's avatar
lm committed
1386
            video2DockWidget->hide();
lm's avatar
lm committed
1387 1388 1389
            break;
        case VIEW_MAVLINK:
            centerStack->setCurrentWidget(protocolWidget);
lm's avatar
lm committed
1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404
            controlDockWidget->hide();
            listDockWidget->hide();
            waypointsDockWidget->hide();
            infoDockWidget->hide();
            debugConsoleDockWidget->hide();
            logPlayerDockWidget->hide();
            mavlinkInspectorWidget->hide();
            parametersDockWidget->hide();
            hsiDockWidget->hide();
            headDown1DockWidget->hide();
            headDown2DockWidget->hide();
            rcViewDockWidget->hide();
            headUpDockWidget->hide();
            video1DockWidget->hide();
            video2DockWidget->hide();
lm's avatar
lm committed
1405 1406
            break;
        case VIEW_OPERATOR:
1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423
            centerStack->setCurrentWidget(mapWidget);
            controlDockWidget->hide();
            listDockWidget->show();
            waypointsDockWidget->show();
            infoDockWidget->hide();
            debugConsoleDockWidget->show();
            logPlayerDockWidget->show();
            parametersDockWidget->hide();
            hsiDockWidget->show();
            headDown1DockWidget->hide();
            headDown2DockWidget->hide();
            rcViewDockWidget->hide();
            headUpDockWidget->show();
            video1DockWidget->hide();
            video2DockWidget->hide();
            mavlinkInspectorWidget->hide();
            break;
lm's avatar
lm committed
1424 1425 1426 1427
        case VIEW_UNCONNECTED:
        case VIEW_FULL:
        default:
            centerStack->setCurrentWidget(mapWidget);
1428
            controlDockWidget->hide();
lm's avatar
lm committed
1429
            listDockWidget->show();
1430 1431
            waypointsDockWidget->hide();
            infoDockWidget->hide();
lm's avatar
lm committed
1432 1433 1434
            debugConsoleDockWidget->show();
            logPlayerDockWidget->show();
            parametersDockWidget->hide();
1435
            hsiDockWidget->hide();
lm's avatar
lm committed
1436 1437 1438 1439 1440 1441
            headDown1DockWidget->hide();
            headDown2DockWidget->hide();
            rcViewDockWidget->hide();
            headUpDockWidget->show();
            video1DockWidget->hide();
            video2DockWidget->hide();
1442
            mavlinkInspectorWidget->show();
lm's avatar
lm committed
1443
            break;
pixhawk's avatar
pixhawk committed
1444 1445 1446
        }
    }

1447 1448 1449 1450
    // Restore the widget positions and size
    if (settings.contains(getWindowStateKey()))
    {
        restoreState(settings.value(getWindowStateKey()).toByteArray(), QGC::applicationVersion());
pixhawk's avatar
pixhawk committed
1451 1452 1453
    }
}

1454
void MainWindow::loadEngineerView()
lm's avatar
lm committed
1455
{
1456 1457
    if (currentView != VIEW_ENGINEER)
    {
1458
        storeViewState();
1459 1460
        currentView = VIEW_ENGINEER;
        ui.actionEngineersView->setChecked(true);
1461
        loadViewState();
1462
    }
lm's avatar
lm committed
1463 1464
}

1465
void MainWindow::loadOperatorView()
lm's avatar
lm committed
1466
{
1467 1468
    if (currentView != VIEW_OPERATOR)
    {
1469
        storeViewState();
1470 1471
        currentView = VIEW_OPERATOR;
        ui.actionOperatorsView->setChecked(true);
1472
        loadViewState();
1473 1474
    }
}
lm's avatar
lm committed
1475

1476 1477
void MainWindow::loadUnconnectedView()
{
1478 1479 1480
    if (currentView != VIEW_UNCONNECTED)
    {
        storeViewState();
1481 1482
        currentView = VIEW_UNCONNECTED;
        ui.actionUnconnectedView->setChecked(true);
1483
        loadViewState();
1484
    }
lm's avatar
lm committed
1485 1486
}

1487
void MainWindow::loadPilotView()
1488
{
1489 1490 1491
    if (currentView != VIEW_PILOT)
    {
        storeViewState();
1492 1493
        currentView = VIEW_PILOT;
        ui.actionPilotsView->setChecked(true);
1494
        loadViewState();
1495
    }
1496 1497
}

1498
void MainWindow::loadMAVLinkView()
1499
{
1500 1501 1502
    if (currentView != VIEW_MAVLINK)
    {
        storeViewState();
1503 1504
        currentView = VIEW_MAVLINK;
        ui.actionMavlinkView->setChecked(true);
1505
        loadViewState();
1506
    }
1507 1508
}

1509 1510
void MainWindow::loadDataView(QString fileName)
{
1511
    // Plot is now selected, now load data from file
1512 1513 1514 1515 1516 1517 1518 1519
    if (dataplotWidget)
    {
        dataplotWidget->loadFile(fileName);
    }
    QStackedWidget *centerStack = dynamic_cast<QStackedWidget*>(centralWidget());
    if (centerStack)
    {
        centerStack->setCurrentWidget(dataplotWidget);
1520
        dataplotWidget->loadFile(fileName);
1521
    }
1522 1523 1524
}


oberion's avatar
oberion committed
1525 1526
QList<QAction*> MainWindow::listLinkMenuActions(void)
{
lm's avatar
lm committed
1527
    return ui.menuNetwork->actions();
1528
}