#include "Waypoint2DIcon.h" #include #include "opmapcontrol.h" #include "QGC.h" Waypoint2DIcon::Waypoint2DIcon(mapcontrol::MapGraphicItem* map, mapcontrol::OPMapWidget* parent, qreal latitude, qreal longitude, qreal altitude, int listindex, QString name, QString description, int radius) : mapcontrol::WayPointItem(internals::PointLatLng(latitude, longitude), altitude, description, map), parent(parent), waypoint(), radius(radius), showAcceptanceRadius(true), showOrbit(false), color(Qt::red) { Q_UNUSED(name); SetHeading(0); SetNumber(listindex); this->setFlag(QGraphicsItem::ItemIgnoresTransformations,true); picture = QPixmap(radius+1, radius+1); autoreachedEnabled = false; // In contrast to the use in OpenPilot, we don't // want to let the map interfere with the actual mission logic // wether a WP is reached depends solely on the UAV's state machine drawIcon(); } Waypoint2DIcon::Waypoint2DIcon(mapcontrol::MapGraphicItem* map, mapcontrol::OPMapWidget* parent, MissionItem* wp, const QColor& color, int listindex, int radius) : mapcontrol::WayPointItem(internals::PointLatLng(wp->getLatitude(), wp->getLongitude()), wp->getAltitude(), wp->getDescription(), map), parent(parent), waypoint(wp), radius(radius), showAcceptanceRadius(true), showOrbit(false), color(color) { SetHeading(wp->getYaw()); SetNumber(listindex); this->setFlag(QGraphicsItem::ItemIgnoresTransformations,true); picture = QPixmap(radius+1, radius+1); autoreachedEnabled = false; // In contrast to the use in OpenPilot, we don't // want to let the map interfere with the actual mission logic // wether a WP is reached depends solely on the UAV's state machine updateWaypoint(); } Waypoint2DIcon::~Waypoint2DIcon() { } void Waypoint2DIcon::SetHeading(float heading) { mapcontrol::WayPointItem::SetHeading(heading); drawIcon(); } void Waypoint2DIcon::updateWaypoint() { if (!waypoint.isNull()) { // Store old size static QRectF oldSize; SetHeading(waypoint->getYaw()); SetCoord(internals::PointLatLng(waypoint->getLatitude(), waypoint->getLongitude())); // qDebug() << "UPDATING WP:" << waypoint->getId() << "LAT:" << waypoint->getLatitude() << "LON:" << waypoint->getLongitude(); SetDescription(waypoint->getDescription()); SetAltitude(waypoint->getAltitude()); // FIXME Add SetNumber (currently needs a separate call) drawIcon(); QRectF newSize = boundingRect(); // qDebug() << "WIDTH" << newSize.width() << "<" << oldSize.width(); // If new size is smaller than old size, update surrounding if ((newSize.width() <= oldSize.width()) || (newSize.height() <= oldSize.height())) { // If the symbol size was reduced, enforce an update of the environment // update(oldSize); int oldWidth = oldSize.width() + 20; int oldHeight = oldSize.height() + 20; map->update(this->x()-10, this->y()-10, oldWidth, oldHeight); //// qDebug() << "UPDATING DUE TO SMALLER SIZE"; //// qDebug() << "X:" << this->x()-1 << "Y:" << this->y()-1 << "WIDTH:" << oldWidth << "HEIGHT:" << oldHeight; } else { // Symbol size stayed constant or increased, use new size for update this->update(); } oldSize = boundingRect(); } } QRectF Waypoint2DIcon::boundingRect() const { int loiter = 0; int acceptance = 0; internals::PointLatLng coord = (internals::PointLatLng)Coord(); if (!waypoint.isNull()) { if (showAcceptanceRadius && (waypoint->getAction() == (int)MAV_CMD_NAV_WAYPOINT)) { acceptance = map->metersToPixels(waypoint->getAcceptanceRadius(), coord); } if (((waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_UNLIM) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TIME) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TURNS))) { loiter = map->metersToPixels(waypoint->getLoiterOrbit(), coord); } } int width = qMax(picture.width()/2, qMax(loiter, acceptance)); int height = qMax(picture.height()/2, qMax(loiter, acceptance)); return QRectF(-width,-height,2*width,2*height); } void Waypoint2DIcon::drawIcon() { picture.fill(Qt::transparent); QPainter painter(&picture); painter.setRenderHint(QPainter::Antialiasing, true); painter.setRenderHint(QPainter::HighQualityAntialiasing, true); QFont font("Bitstream Vera Sans"); int fontSize = picture.height()*0.8f; font.setPixelSize(fontSize); QFontMetrics metrics = QFontMetrics(font); int border = qMax(4, metrics.leading()); painter.setFont(font); painter.setRenderHint(QPainter::TextAntialiasing); QPen pen1(Qt::black); pen1.setWidth(4); QPen pen2(color); pen2.setWidth(2); painter.setBrush(Qt::NoBrush); int penWidth = pen1.width(); // DRAW WAYPOINT QPointF p(picture.width()/2, picture.height()/2); QPolygonF poly(4); // Top point poly.replace(0, QPointF(p.x(), p.y()-picture.height()/2.0f+penWidth/2)); // Right point poly.replace(1, QPointF(p.x()+picture.width()/2.0f-penWidth/2, p.y())); // Bottom point poly.replace(2, QPointF(p.x(), p.y() + picture.height()/2.0f-penWidth/2)); poly.replace(3, QPointF(p.x() - picture.width()/2.0f+penWidth/2, p.y())); int waypointSize = qMin(picture.width(), picture.height()); float rad = (waypointSize/2.0f) * 0.7f * (1/sqrt(2.0f)); // If this is not a waypoint (only the default representation) // or it is a waypoint, but not one where direction has no meaning // then draw the heading indicator if (waypoint.isNull() || (waypoint && ( (waypoint->getAction() != (int)MAV_CMD_NAV_TAKEOFF) && (waypoint->getAction() != (int)MAV_CMD_NAV_LAND) && (waypoint->getAction() != (int)MAV_CMD_NAV_LOITER_UNLIM) && (waypoint->getAction() != (int)MAV_CMD_NAV_LOITER_TIME) && (waypoint->getAction() != (int)MAV_CMD_NAV_LOITER_TURNS) && (waypoint->getAction() != (int)MAV_CMD_NAV_RETURN_TO_LAUNCH) ))) { painter.setPen(pen1); painter.drawLine(p.x(), p.y(), p.x()+sin(Heading()/180.0f*M_PI) * rad, p.y()-cos(Heading()/180.0f*M_PI) * rad); painter.setPen(pen2); painter.drawLine(p.x(), p.y(), p.x()+sin(Heading()/180.0f*M_PI) * rad, p.y()-cos(Heading()/180.0f*M_PI) * rad); } if (((!waypoint.isNull())) && (waypoint->getAction() == (int)MAV_CMD_NAV_TAKEOFF)) { // Takeoff waypoint int width = picture.width()-penWidth; int height = picture.height()-penWidth; painter.setPen(pen1); painter.drawRect(penWidth/2, penWidth/2, width, height); painter.setPen(pen2); painter.drawRect(penWidth/2, penWidth/2, width, height); painter.setPen(pen1); painter.drawRect(width*0.3, height*0.3f, width*0.6f, height*0.6f); painter.setPen(pen2); painter.drawRect(width*0.3, height*0.3f, width*0.6f, height*0.6f); } else if (((!waypoint.isNull())) && (waypoint->getAction() == (int)MAV_CMD_NAV_LAND)) { // Landing waypoint int width = (picture.width())/2-penWidth; int height = (picture.height())/2-penWidth; painter.setPen(pen1); painter.drawEllipse(p, width, height); painter.drawLine(p.x()-width/2, p.y()-height/2, 2*width, 2*height); painter.setPen(pen2); painter.drawEllipse(p, width, height); painter.drawLine(p.x()-width/2, p.y()-height/2, 2*width, 2*height); } else if (((!waypoint.isNull())) && ((waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_UNLIM) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TIME) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TURNS))) { // Loiter waypoint int width = (picture.width()-penWidth)/2; int height = (picture.height()-penWidth)/2; painter.setPen(pen1); painter.drawEllipse(p, width, height); painter.drawPoint(p); painter.setPen(pen2); painter.drawEllipse(p, width, height); painter.drawPoint(p); } else if (((!waypoint.isNull())) && (waypoint->getAction() == (int)MAV_CMD_NAV_RETURN_TO_LAUNCH)) { // Return to launch waypoint int width = picture.width()-penWidth; int height = picture.height()-penWidth; painter.setPen(pen1); painter.drawRect(penWidth/2, penWidth/2, width, height); painter.setPen(pen2); painter.drawRect(penWidth/2, penWidth/2, width, height); QString text("R"); painter.setPen(pen1); QRect rect = metrics.boundingRect(0, 0, width - 2*border, height, Qt::AlignLeft | Qt::TextWordWrap, text); painter.drawText(width/4, height/6, rect.width(), rect.height(), Qt::AlignCenter | Qt::TextWordWrap, text); painter.setPen(pen2); font.setPixelSize(fontSize*0.85f); painter.setFont(font); painter.drawText(width/4, height/6, rect.width(), rect.height(), Qt::AlignCenter | Qt::TextWordWrap, text); } else { // Navigation waypoint painter.setPen(pen1); painter.drawPolygon(poly); painter.setPen(pen2); painter.drawPolygon(poly); } } void Waypoint2DIcon::SetShowNumber(const bool &value) { shownumber=value; if((numberI==0) && value) { numberI=new QGraphicsSimpleTextItem(this); numberIBG=new QGraphicsRectItem(this); numberIBG->setBrush(Qt::black); numberIBG->setOpacity(0.5); numberI->setZValue(3); numberI->setPen(QPen(QGC::colorCyan)); numberI->setPos(5,-picture.height()); numberIBG->setPos(5,-picture.height()); numberI->setText(QString::number(number)); numberIBG->setRect(numberI->boundingRect().adjusted(-2,0,1,0)); } else if (!value && numberI) { delete numberI; delete numberIBG; } this->update(); } void Waypoint2DIcon::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) { Q_UNUSED(option); Q_UNUSED(widget); QPen pen = painter->pen(); pen.setWidth(2); painter->drawPixmap(-picture.width()/2,-picture.height()/2,picture); if (this->isSelected()) { pen.setColor(Qt::yellow); painter->drawRect(QRectF(-picture.width()/2,-picture.height()/2,picture.width()-1,picture.height()-1)); } QPen penBlack(Qt::black); penBlack.setWidth(4); pen.setColor(color); if ((!waypoint.isNull()) && (waypoint->getAction() == (int)MAV_CMD_NAV_WAYPOINT) && showAcceptanceRadius) { QPen redPen = QPen(pen); redPen.setColor(Qt::yellow); redPen.setWidth(1); painter->setPen(redPen); const int acceptance = map->metersToPixels(waypoint->getAcceptanceRadius(), Coord()); if (acceptance > 0) { painter->setPen(penBlack); painter->drawEllipse(QPointF(0, 0), acceptance, acceptance); painter->setPen(redPen); painter->drawEllipse(QPointF(0, 0), acceptance, acceptance); } } if ((!waypoint.isNull()) && ((waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_UNLIM) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TIME) || (waypoint->getAction() == (int)MAV_CMD_NAV_LOITER_TURNS))) { QPen penDash(color); penDash.setWidth(1); //penDash.setStyle(Qt::DotLine); // A negative radius indicates counter-clockwise rotation, but we still want to draw it positive const int loiter = map->metersToPixels(fabs(waypoint->getLoiterOrbit()), Coord()); if (loiter > picture.width()/2) { painter->setPen(penBlack); painter->drawEllipse(QPointF(0, 0), loiter, loiter); painter->setPen(penDash); painter->drawEllipse(QPointF(0, 0), loiter, loiter); } } }