#include "selfdrive/ui/qt/maps/map_instructions.h" #include #include #include "selfdrive/ui/qt/maps/map_helpers.h" #include "selfdrive/ui/ui.h" const QString ICON_SUFFIX = ".png"; MapInstructions::MapInstructions(QWidget *parent) : QWidget(parent) { is_rhd = Params().getBool("IsRhdDetected"); QVBoxLayout *main_layout = new QVBoxLayout(this); main_layout->setContentsMargins(11, UI_BORDER_SIZE, 11, 20); QHBoxLayout *top_layout = new QHBoxLayout; top_layout->addWidget(icon_01 = new QLabel, 0, Qt::AlignTop); QVBoxLayout *right_layout = new QVBoxLayout; right_layout->setContentsMargins(9, 9, 9, 0); right_layout->addWidget(distance = new QLabel); distance->setStyleSheet(R"(font-size: 90px;)"); right_layout->addWidget(primary = new QLabel); primary->setStyleSheet(R"(font-size: 60px;)"); primary->setWordWrap(true); right_layout->addWidget(secondary = new QLabel); secondary->setStyleSheet(R"(font-size: 50px;)"); secondary->setWordWrap(true); top_layout->addLayout(right_layout); main_layout->addLayout(top_layout); main_layout->addLayout(lane_layout = new QHBoxLayout); lane_layout->setAlignment(Qt::AlignHCenter); lane_layout->setSpacing(10); setStyleSheet("color:white"); QPalette pal = palette(); pal.setColor(QPalette::Background, QColor(0, 0, 0, 150)); setAutoFillBackground(true); setPalette(pal); buildPixmapCache(); } void MapInstructions::buildPixmapCache() { QDir dir("../assets/navigation"); for (QString fn : dir.entryList({"*" + ICON_SUFFIX}, QDir::Files)) { QPixmap pm(dir.filePath(fn)); QString key = fn.left(fn.size() - ICON_SUFFIX.length()); pm = pm.scaledToWidth(200, Qt::SmoothTransformation); // Maneuver icons pixmap_cache[key] = pm; // lane direction icons if (key.contains("turn_")) { pixmap_cache["lane_" + key] = pm.scaled({125, 125}, Qt::IgnoreAspectRatio, Qt::SmoothTransformation); } // for rhd, reflect direction and then flip if (key.contains("_left")) { pixmap_cache["rhd_" + key.replace("_left", "_right")] = pm.transformed(QTransform().scale(-1, 1)); } else if (key.contains("_right")) { pixmap_cache["rhd_" + key.replace("_right", "_left")] = pm.transformed(QTransform().scale(-1, 1)); } } } void MapInstructions::updateInstructions(cereal::NavInstruction::Reader instruction) { setUpdatesEnabled(false); // Show instruction text QString primary_str = QString::fromStdString(instruction.getManeuverPrimaryText()); QString secondary_str = QString::fromStdString(instruction.getManeuverSecondaryText()); primary->setText(primary_str); secondary->setVisible(secondary_str.length() > 0); secondary->setText(secondary_str); auto distance_str_pair = map_format_distance(instruction.getManeuverDistance(), uiState()->scene.is_metric); distance->setText(QString("%1 %2").arg(distance_str_pair.first, distance_str_pair.second)); // Show arrow with direction QString type = QString::fromStdString(instruction.getManeuverType()); QString modifier = QString::fromStdString(instruction.getManeuverModifier()); if (!type.isEmpty()) { QString fn = "direction_" + type; if (!modifier.isEmpty()) { fn += "_" + modifier; } fn = fn.replace(' ', '_'); bool rhd = is_rhd && (fn.contains("_left") || fn.contains("_right")); icon_01->setPixmap(pixmap_cache[!rhd ? fn : "rhd_" + fn]); icon_01->setSizePolicy(QSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed)); icon_01->setVisible(true); } else { icon_01->setVisible(false); } // Hide distance after arrival distance->setVisible(type != "arrive" || instruction.getManeuverDistance() > 0); // Show lanes auto lanes = instruction.getLanes(); for (int i = 0; i < lanes.size(); ++i) { bool active = lanes[i].getActive(); const auto active_direction = lanes[i].getActiveDirection(); // TODO: Make more images based on active direction and combined directions QString fn = "lane_direction_"; // active direction has precedence if (active && active_direction != cereal::NavInstruction::Direction::NONE) { fn += "turn_" + DIRECTIONS[active_direction]; } else { for (auto const &direction : lanes[i].getDirections()) { if (direction != cereal::NavInstruction::Direction::NONE) { fn += "turn_" + DIRECTIONS[direction]; break; } } } if (!active) { fn += "_inactive"; } QLabel *label = (i < lane_labels.size()) ? lane_labels[i] : lane_labels.emplace_back(new QLabel); if (!label->parentWidget()) { lane_layout->addWidget(label); } label->setPixmap(pixmap_cache[fn]); label->setVisible(true); } for (int i = lanes.size(); i < lane_labels.size(); ++i) { lane_labels[i]->setVisible(false); } setUpdatesEnabled(true); setVisible(true); }