|  |  |  | #include "selfdrive/ui/qt/sidebar.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QMouseEvent>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "selfdrive/ui/qt/util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Sidebar::drawMetric(QPainter &p, const QPair<QString, QString> &label, QColor c, int y) {
 | 
					
						
							|  |  |  |   const QRect rect = {30, y, 240, 126};
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p.setPen(Qt::NoPen);
 | 
					
						
							|  |  |  |   p.setBrush(QBrush(c));
 | 
					
						
							|  |  |  |   p.setClipRect(rect.x() + 4, rect.y(), 18, rect.height(), Qt::ClipOperation::ReplaceClip);
 | 
					
						
							|  |  |  |   p.drawRoundedRect(QRect(rect.x() + 4, rect.y() + 4, 100, 118), 18, 18);
 | 
					
						
							|  |  |  |   p.setClipping(false);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QPen pen = QPen(QColor(0xff, 0xff, 0xff, 0x55));
 | 
					
						
							|  |  |  |   pen.setWidth(2);
 | 
					
						
							|  |  |  |   p.setPen(pen);
 | 
					
						
							|  |  |  |   p.setBrush(Qt::NoBrush);
 | 
					
						
							|  |  |  |   p.drawRoundedRect(rect, 20, 20);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p.setPen(QColor(0xff, 0xff, 0xff));
 | 
					
						
							|  |  |  |   configFont(p, "Inter", 35, "SemiBold");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QRect label_rect = getTextRect(p, Qt::AlignCenter, label.first);
 | 
					
						
							|  |  |  |   label_rect.setWidth(218);
 | 
					
						
							|  |  |  |   label_rect.moveLeft(rect.left() + 22);
 | 
					
						
							|  |  |  |   label_rect.moveTop(rect.top() + 19);
 | 
					
						
							|  |  |  |   p.drawText(label_rect, Qt::AlignCenter, label.first);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   label_rect.moveTop(rect.top() + 65);
 | 
					
						
							|  |  |  |   p.drawText(label_rect, Qt::AlignCenter, label.second);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Sidebar::Sidebar(QWidget *parent) : QFrame(parent) {
 | 
					
						
							|  |  |  |   home_img = loadPixmap("../assets/images/button_home.png", {180, 180});
 | 
					
						
							|  |  |  |   settings_img = loadPixmap("../assets/images/button_settings.png", settings_btn.size(), Qt::IgnoreAspectRatio);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   connect(this, &Sidebar::valueChanged, [=] { update(); });
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   setAttribute(Qt::WA_OpaquePaintEvent);
 | 
					
						
							|  |  |  |   setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding);
 | 
					
						
							|  |  |  |   setFixedWidth(300);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   QObject::connect(uiState(), &UIState::uiUpdate, this, &Sidebar::updateState);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Sidebar::mouseReleaseEvent(QMouseEvent *event) {
 | 
					
						
							|  |  |  |   if (settings_btn.contains(event->pos())) {
 | 
					
						
							|  |  |  |     emit openSettings();
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Sidebar::updateState(const UIState &s) {
 | 
					
						
							|  |  |  |   if (!isVisible()) return;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto &sm = *(s.sm);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto deviceState = sm["deviceState"].getDeviceState();
 | 
					
						
							|  |  |  |   setProperty("netType", network_type[deviceState.getNetworkType()]);
 | 
					
						
							|  |  |  |   int strength = (int)deviceState.getNetworkStrength();
 | 
					
						
							|  |  |  |   setProperty("netStrength", strength > 0 ? strength + 1 : 0);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ItemStatus connectStatus;
 | 
					
						
							|  |  |  |   auto last_ping = deviceState.getLastAthenaPingTime();
 | 
					
						
							|  |  |  |   if (last_ping == 0) {
 | 
					
						
							|  |  |  |     connectStatus = ItemStatus{{tr("CONNECT"), tr("OFFLINE")}, warning_color};
 | 
					
						
							|  |  |  |   } else {
 | 
					
						
							|  |  |  |     connectStatus = nanos_since_boot() - last_ping < 80e9 ? ItemStatus{{tr("CONNECT"), tr("ONLINE")}, good_color} : ItemStatus{{tr("CONNECT"), tr("ERROR")}, danger_color};
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   setProperty("connectStatus", QVariant::fromValue(connectStatus));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ItemStatus tempStatus = {{tr("TEMP"), tr("HIGH")}, danger_color};
 | 
					
						
							|  |  |  |   auto ts = deviceState.getThermalStatus();
 | 
					
						
							|  |  |  |   if (ts == cereal::DeviceState::ThermalStatus::GREEN) {
 | 
					
						
							|  |  |  |     tempStatus = {{tr("TEMP"), tr("GOOD")}, good_color};
 | 
					
						
							|  |  |  |   } else if (ts == cereal::DeviceState::ThermalStatus::YELLOW) {
 | 
					
						
							|  |  |  |     tempStatus = {{tr("TEMP"), tr("OK")}, warning_color};
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   setProperty("tempStatus", QVariant::fromValue(tempStatus));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ItemStatus pandaStatus = {{tr("VEHICLE"), tr("ONLINE")}, good_color};
 | 
					
						
							|  |  |  |   if (s.scene.pandaType == cereal::PandaState::PandaType::UNKNOWN) {
 | 
					
						
							|  |  |  |     pandaStatus = {{tr("NO"), tr("PANDA")}, danger_color};
 | 
					
						
							|  |  |  |   } else if (s.scene.started && !sm["liveLocationKalman"].getLiveLocationKalman().getGpsOK()) {
 | 
					
						
							|  |  |  |     pandaStatus = {{tr("GPS"), tr("SEARCH")}, warning_color};
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  |   setProperty("pandaStatus", QVariant::fromValue(pandaStatus));
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Sidebar::paintEvent(QPaintEvent *event) {
 | 
					
						
							|  |  |  |   QPainter p(this);
 | 
					
						
							|  |  |  |   p.setPen(Qt::NoPen);
 | 
					
						
							|  |  |  |   p.setRenderHint(QPainter::Antialiasing);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p.fillRect(rect(), QColor(57, 57, 57));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // static imgs
 | 
					
						
							|  |  |  |   p.setOpacity(0.65);
 | 
					
						
							|  |  |  |   p.drawPixmap(settings_btn.x(), settings_btn.y(), settings_img);
 | 
					
						
							|  |  |  |   p.setOpacity(1.0);
 | 
					
						
							|  |  |  |   p.drawPixmap(60, 1080 - 180 - 40, home_img);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // network
 | 
					
						
							|  |  |  |   int x = 58;
 | 
					
						
							|  |  |  |   const QColor gray(0x54, 0x54, 0x54);
 | 
					
						
							|  |  |  |   for (int i = 0; i < 5; ++i) {
 | 
					
						
							|  |  |  |     p.setBrush(i < net_strength ? Qt::white : gray);
 | 
					
						
							|  |  |  |     p.drawEllipse(x, 196, 27, 27);
 | 
					
						
							|  |  |  |     x += 37;
 | 
					
						
							|  |  |  |   }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   configFont(p, "Inter", 35, "Regular");
 | 
					
						
							|  |  |  |   p.setPen(QColor(0xff, 0xff, 0xff));
 | 
					
						
							|  |  |  |   const QRect r = QRect(50, 247, 100, 50);
 | 
					
						
							|  |  |  |   p.drawText(r, Qt::AlignCenter, net_type);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // metrics
 | 
					
						
							|  |  |  |   drawMetric(p, temp_status.first, temp_status.second, 338);
 | 
					
						
							|  |  |  |   drawMetric(p, panda_status.first, panda_status.second, 496);
 | 
					
						
							|  |  |  |   drawMetric(p, connect_status.first, connect_status.second, 654);
 | 
					
						
							|  |  |  | }
 |