2011-03-13 16:08:02 +01:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
*
|
|
|
|
* @file systemhealthgadgetwidget.cpp
|
2012-07-02 17:54:21 +02:00
|
|
|
* @author OpenPilot Team & Edouard Lafargue Copyright (C) 2012.
|
2011-03-13 16:08:02 +01:00
|
|
|
* @addtogroup GCSPlugins GCS Plugins
|
|
|
|
* @{
|
|
|
|
* @addtogroup SystemHealthPlugin System Health Plugin
|
|
|
|
* @{
|
|
|
|
* @brief The System Health gadget plugin
|
|
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program 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 this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "systemhealthgadgetwidget.h"
|
|
|
|
#include "utils/stylehelper.h"
|
|
|
|
#include "extensionsystem/pluginmanager.h"
|
|
|
|
#include "uavobjectmanager.h"
|
|
|
|
#include "systemalarms.h"
|
|
|
|
|
|
|
|
#include <QDebug>
|
2012-06-24 02:22:43 +02:00
|
|
|
#include <QWhatsThis>
|
2011-03-13 16:08:02 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the widget
|
|
|
|
*/
|
|
|
|
SystemHealthGadgetWidget::SystemHealthGadgetWidget(QWidget *parent) : QGraphicsView(parent)
|
|
|
|
{
|
2013-05-19 16:37:30 +02:00
|
|
|
setMinimumSize(128, 128);
|
2011-03-13 16:08:02 +01:00
|
|
|
setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
|
|
|
setScene(new QGraphicsScene(this));
|
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
|
2011-03-13 16:08:02 +01:00
|
|
|
m_renderer = new QSvgRenderer();
|
|
|
|
background = new QGraphicsSvgItem();
|
|
|
|
foreground = new QGraphicsSvgItem();
|
2013-05-19 16:37:30 +02:00
|
|
|
nolink = new QGraphicsSvgItem();
|
2013-05-21 00:15:18 +02:00
|
|
|
missingElements = new QStringList();
|
2011-03-13 16:08:02 +01:00
|
|
|
paint();
|
|
|
|
|
|
|
|
// Now connect the widget to the SystemAlarms UAVObject
|
|
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
|
|
UAVObjectManager *objManager = pm->getObject<UAVObjectManager>();
|
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
SystemAlarms *obj = dynamic_cast<SystemAlarms *>(objManager->getObject(QString("SystemAlarms")));
|
|
|
|
connect(obj, SIGNAL(objectUpdated(UAVObject *)), this, SLOT(updateAlarms(UAVObject *)));
|
2011-03-13 16:08:02 +01:00
|
|
|
|
|
|
|
// Listen to autopilot connection events
|
2013-05-19 16:37:30 +02:00
|
|
|
TelemetryManager *telMngr = pm->getObject<TelemetryManager>();
|
2011-03-13 16:08:02 +01:00
|
|
|
connect(telMngr, SIGNAL(connected()), this, SLOT(onAutopilotConnect()));
|
|
|
|
connect(telMngr, SIGNAL(disconnected()), this, SLOT(onAutopilotDisconnect()));
|
|
|
|
|
2012-07-09 00:04:39 +02:00
|
|
|
setToolTip(tr("Displays flight system errors. Click on an alarm for more information."));
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-05-19 16:37:30 +02:00
|
|
|
* Hide the "No Link" overlay
|
|
|
|
*/
|
2011-03-13 16:08:02 +01:00
|
|
|
void SystemHealthGadgetWidget::onAutopilotConnect()
|
|
|
|
{
|
|
|
|
nolink->setVisible(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-05-19 16:37:30 +02:00
|
|
|
* Show the "No Link" overlay
|
|
|
|
*/
|
2011-03-13 16:08:02 +01:00
|
|
|
void SystemHealthGadgetWidget::onAutopilotDisconnect()
|
|
|
|
{
|
|
|
|
nolink->setVisible(true);
|
|
|
|
}
|
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
void SystemHealthGadgetWidget::updateAlarms(UAVObject *systemAlarm)
|
2011-03-13 16:08:02 +01:00
|
|
|
{
|
|
|
|
// This code does not know anything about alarms beforehand, and
|
|
|
|
// I found no efficient way to locate items inside the scene by
|
|
|
|
// name, so it's just as simple to reset the scene:
|
|
|
|
// And add the one with the right name.
|
|
|
|
QGraphicsScene *m_scene = scene();
|
2013-05-19 16:37:30 +02:00
|
|
|
|
|
|
|
foreach(QGraphicsItem * item, background->childItems()) {
|
2011-03-13 16:08:02 +01:00
|
|
|
m_scene->removeItem(item);
|
|
|
|
delete item; // removeItem does _not_ delete the item.
|
|
|
|
}
|
|
|
|
|
|
|
|
QString alarm = systemAlarm->getName();
|
2013-05-19 16:37:30 +02:00
|
|
|
foreach(UAVObjectField * field, systemAlarm->getFields()) {
|
2011-03-13 16:08:02 +01:00
|
|
|
for (uint i = 0; i < field->getNumElements(); ++i) {
|
|
|
|
QString element = field->getElementNames()[i];
|
2013-05-19 16:37:30 +02:00
|
|
|
QString value = field->getValue(i).toString();
|
2013-05-21 00:15:18 +02:00
|
|
|
if (!missingElements->contains(element)){
|
|
|
|
if (m_renderer->elementExists(element)) {
|
|
|
|
QMatrix blockMatrix = m_renderer->matrixForElement(element);
|
|
|
|
qreal startX = blockMatrix.mapRect(m_renderer->boundsOnElement(element)).x();
|
|
|
|
qreal startY = blockMatrix.mapRect(m_renderer->boundsOnElement(element)).y();
|
|
|
|
QString element2 = element + "-" + value;
|
|
|
|
if (!missingElements->contains(element2)) {
|
|
|
|
if (m_renderer->elementExists(element2)) {
|
|
|
|
QGraphicsSvgItem *ind = new QGraphicsSvgItem();
|
|
|
|
ind->setSharedRenderer(m_renderer);
|
|
|
|
ind->setElementId(element2);
|
|
|
|
ind->setParentItem(background);
|
|
|
|
QTransform matrix;
|
|
|
|
matrix.translate(startX, startY);
|
|
|
|
ind->setTransform(matrix, false);
|
|
|
|
} else {
|
|
|
|
if (value.compare("Uninitialised") != 0) {
|
|
|
|
missingElements->append(element2);
|
|
|
|
qDebug() << "Warning: element " << element2 << " not found in SVG.";
|
|
|
|
}
|
|
|
|
}
|
2013-05-19 16:37:30 +02:00
|
|
|
}
|
2013-05-21 00:15:18 +02:00
|
|
|
} else {
|
|
|
|
missingElements->append(element);
|
|
|
|
qDebug() << "Warning: Element " << element << " not found in SVG.";
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SystemHealthGadgetWidget::~SystemHealthGadgetWidget()
|
|
|
|
{
|
2013-05-19 16:37:30 +02:00
|
|
|
// Do nothing
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SystemHealthGadgetWidget::setSystemFile(QString dfn)
|
|
|
|
{
|
2013-05-21 00:15:18 +02:00
|
|
|
// Clear the list of elements not found on svg
|
|
|
|
missingElements->clear();
|
2011-03-13 16:08:02 +01:00
|
|
|
setBackgroundBrush(QBrush(Utils::StyleHelper::baseColor()));
|
2013-05-19 16:37:30 +02:00
|
|
|
if (QFile::exists(dfn)) {
|
|
|
|
m_renderer->load(dfn);
|
|
|
|
if (m_renderer->isValid()) {
|
|
|
|
fgenabled = false;
|
|
|
|
background->setSharedRenderer(m_renderer);
|
|
|
|
background->setElementId("background");
|
|
|
|
|
|
|
|
if (m_renderer->elementExists("foreground")) {
|
|
|
|
foreground->setSharedRenderer(m_renderer);
|
|
|
|
foreground->setElementId("foreground");
|
|
|
|
foreground->setZValue(99);
|
|
|
|
fgenabled = true;
|
|
|
|
}
|
|
|
|
if (m_renderer->elementExists("nolink")) {
|
|
|
|
nolink->setSharedRenderer(m_renderer);
|
|
|
|
nolink->setElementId("nolink");
|
|
|
|
nolink->setZValue(100);
|
|
|
|
}
|
|
|
|
|
|
|
|
QGraphicsScene *l_scene = scene();
|
|
|
|
l_scene->setSceneRect(background->boundingRect());
|
|
|
|
fitInView(background, Qt::KeepAspectRatio);
|
|
|
|
|
|
|
|
// Check whether the autopilot is connected already, by the way:
|
|
|
|
ExtensionSystem::PluginManager *pm = ExtensionSystem::PluginManager::instance();
|
|
|
|
UAVObjectManager *objManager = pm->getObject<UAVObjectManager>();
|
|
|
|
TelemetryManager *telMngr = pm->getObject<TelemetryManager>();
|
|
|
|
if (telMngr->isConnected()) {
|
|
|
|
onAutopilotConnect();
|
|
|
|
SystemAlarms *obj = dynamic_cast<SystemAlarms *>(objManager->getObject(QString("SystemAlarms")));
|
|
|
|
updateAlarms(obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else { qDebug() << "SystemHealthGadget: no file"; }
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void SystemHealthGadgetWidget::paint()
|
|
|
|
{
|
|
|
|
QGraphicsScene *l_scene = scene();
|
2013-05-19 16:37:30 +02:00
|
|
|
|
2011-03-13 16:08:02 +01:00
|
|
|
l_scene->clear();
|
|
|
|
l_scene->addItem(background);
|
|
|
|
l_scene->addItem(foreground);
|
|
|
|
l_scene->addItem(nolink);
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SystemHealthGadgetWidget::paintEvent(QPaintEvent *event)
|
|
|
|
{
|
|
|
|
// Skip painting until the dial file is loaded
|
2013-05-19 16:37:30 +02:00
|
|
|
if (!m_renderer->isValid()) {
|
|
|
|
qDebug() << "SystemHealthGadget: System file not loaded, not rendering";
|
2011-03-13 16:08:02 +01:00
|
|
|
return;
|
|
|
|
}
|
2013-05-19 16:37:30 +02:00
|
|
|
QGraphicsView::paintEvent(event);
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// This event enables the dial to be dynamically resized
|
|
|
|
// whenever the gadget is resized, taking advantage of the vector
|
|
|
|
// nature of SVG dials.
|
|
|
|
void SystemHealthGadgetWidget::resizeEvent(QResizeEvent *event)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event);
|
2013-05-19 16:37:30 +02:00
|
|
|
fitInView(background, Qt::KeepAspectRatio);
|
2011-03-13 16:08:02 +01:00
|
|
|
}
|
2012-06-24 02:22:43 +02:00
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
void SystemHealthGadgetWidget::mousePressEvent(QMouseEvent *event)
|
2012-06-24 02:22:43 +02:00
|
|
|
{
|
|
|
|
QGraphicsScene *graphicsScene = scene();
|
2013-05-19 16:37:30 +02:00
|
|
|
|
|
|
|
if (graphicsScene) {
|
2012-06-24 02:22:43 +02:00
|
|
|
QPoint point = event->pos();
|
2012-07-10 00:42:59 +02:00
|
|
|
bool haveAlarmItem = false;
|
2013-05-19 16:37:30 +02:00
|
|
|
foreach(QGraphicsItem * sceneItem, items(point)) {
|
|
|
|
QGraphicsSvgItem *clickedItem = dynamic_cast<QGraphicsSvgItem *>(sceneItem);
|
2012-06-24 02:22:43 +02:00
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
if (clickedItem) {
|
|
|
|
if ((clickedItem != foreground) && (clickedItem != background)) {
|
2012-07-10 00:42:59 +02:00
|
|
|
// Clicked an actual alarm. We need to set haveAlarmItem to true
|
|
|
|
// as two of the items in this loop will always be foreground and
|
|
|
|
// background. Without this flag, at some point in the loop we
|
|
|
|
// would always call showAllAlarmDescriptions...
|
|
|
|
haveAlarmItem = true;
|
|
|
|
QString itemId = clickedItem->elementId();
|
2013-05-19 16:37:30 +02:00
|
|
|
if (itemId.contains("OK")) {
|
2012-07-10 00:42:59 +02:00
|
|
|
// No alarm set for this item
|
|
|
|
showAlarmDescriptionForItemId("AlarmOK", event->globalPos());
|
2013-05-19 16:37:30 +02:00
|
|
|
} else {
|
2012-07-10 00:42:59 +02:00
|
|
|
// Warning, error or critical alarm
|
|
|
|
showAlarmDescriptionForItemId(itemId, event->globalPos());
|
|
|
|
}
|
2013-05-19 16:37:30 +02:00
|
|
|
} else if (!haveAlarmItem) {
|
2012-07-10 00:42:59 +02:00
|
|
|
// Clicked foreground or background
|
|
|
|
showAllAlarmDescriptions(event->globalPos());
|
2012-06-26 23:14:56 +02:00
|
|
|
}
|
2012-06-24 02:22:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-07-09 00:04:39 +02:00
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
void SystemHealthGadgetWidget::showAlarmDescriptionForItemId(const QString itemId, const QPoint & location)
|
|
|
|
{
|
2012-07-09 00:04:39 +02:00
|
|
|
QFile alarmDescription(":/systemhealth/html/" + itemId + ".html");
|
2013-05-19 16:37:30 +02:00
|
|
|
|
|
|
|
if (alarmDescription.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
2012-07-09 00:04:39 +02:00
|
|
|
QTextStream textStream(&alarmDescription);
|
|
|
|
QWhatsThis::showText(location, textStream.readAll());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-19 16:37:30 +02:00
|
|
|
void SystemHealthGadgetWidget::showAllAlarmDescriptions(const QPoint & location)
|
|
|
|
{
|
2012-07-10 00:42:59 +02:00
|
|
|
QGraphicsScene *graphicsScene = scene();
|
2013-05-19 16:37:30 +02:00
|
|
|
|
|
|
|
if (graphicsScene) {
|
2012-07-10 00:42:59 +02:00
|
|
|
QString alarmsText;
|
|
|
|
|
|
|
|
// Loop through all items in the scene looking for svg items that represent alarms
|
2013-05-19 16:37:30 +02:00
|
|
|
foreach(QGraphicsItem * curItem, graphicsScene->items()) {
|
|
|
|
QGraphicsSvgItem *curSvgItem = dynamic_cast<QGraphicsSvgItem *>(curItem);
|
|
|
|
|
|
|
|
if (curSvgItem && (curSvgItem != foreground) && (curSvgItem != background)) {
|
2012-07-10 00:42:59 +02:00
|
|
|
QString elementId = curSvgItem->elementId();
|
2013-05-19 16:37:30 +02:00
|
|
|
if (!elementId.contains("OK")) {
|
2012-07-10 00:42:59 +02:00
|
|
|
// Found an alarm, get its corresponding alarm html file contents
|
|
|
|
// and append to the cumulative string for all alarms.
|
|
|
|
QFile alarmDescription(":/systemhealth/html/" + elementId + ".html");
|
2013-05-19 16:37:30 +02:00
|
|
|
if (alarmDescription.open(QIODevice::ReadOnly | QIODevice::Text)) {
|
2012-07-10 00:42:59 +02:00
|
|
|
QTextStream textStream(&alarmDescription);
|
|
|
|
alarmsText.append(textStream.readAll());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-07-09 00:04:39 +02:00
|
|
|
|
2012-07-10 00:42:59 +02:00
|
|
|
// Show alarms text if we have any
|
2013-05-19 16:37:30 +02:00
|
|
|
if (alarmsText.length() > 0) {
|
2012-07-10 00:42:59 +02:00
|
|
|
QWhatsThis::showText(location, alarmsText);
|
|
|
|
}
|
|
|
|
}
|
2012-07-09 00:04:39 +02:00
|
|
|
}
|