You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
99 lines
3.6 KiB
99 lines
3.6 KiB
#include "selfdrive/ui/qt/offroad/driverview.h"
|
|
|
|
#include <QPainter>
|
|
|
|
#include "selfdrive/ui/qt/qt_window.h"
|
|
#include "selfdrive/ui/qt/util.h"
|
|
|
|
const int FACE_IMG_SIZE = 130;
|
|
|
|
DriverViewWindow::DriverViewWindow(QWidget* parent) : QWidget(parent) {
|
|
setAttribute(Qt::WA_OpaquePaintEvent);
|
|
layout = new QStackedLayout(this);
|
|
layout->setStackingMode(QStackedLayout::StackAll);
|
|
|
|
cameraView = new CameraViewWidget("camerad", VISION_STREAM_DRIVER, true, this);
|
|
layout->addWidget(cameraView);
|
|
|
|
scene = new DriverViewScene(this);
|
|
connect(cameraView, &CameraViewWidget::vipcThreadFrameReceived, scene, &DriverViewScene::frameUpdated);
|
|
layout->addWidget(scene);
|
|
layout->setCurrentWidget(scene);
|
|
}
|
|
|
|
void DriverViewWindow::mouseReleaseEvent(QMouseEvent* e) {
|
|
emit done();
|
|
}
|
|
|
|
DriverViewScene::DriverViewScene(QWidget* parent) : sm({"driverState"}), QWidget(parent) {
|
|
face_img = loadPixmap("../assets/img_driver_face.png", {FACE_IMG_SIZE, FACE_IMG_SIZE});
|
|
}
|
|
|
|
void DriverViewScene::showEvent(QShowEvent* event) {
|
|
frame_updated = false;
|
|
is_rhd = params.getBool("IsRHD");
|
|
params.putBool("IsDriverViewEnabled", true);
|
|
}
|
|
|
|
void DriverViewScene::hideEvent(QHideEvent* event) {
|
|
params.putBool("IsDriverViewEnabled", false);
|
|
}
|
|
|
|
void DriverViewScene::frameUpdated() {
|
|
frame_updated = true;
|
|
sm.update(0);
|
|
update();
|
|
}
|
|
|
|
void DriverViewScene::paintEvent(QPaintEvent* event) {
|
|
QPainter p(this);
|
|
|
|
// startup msg
|
|
if (!frame_updated) {
|
|
p.setPen(Qt::white);
|
|
p.setRenderHint(QPainter::TextAntialiasing);
|
|
configFont(p, "Inter", 100, "Bold");
|
|
p.drawText(geometry(), Qt::AlignCenter, "camera starting");
|
|
return;
|
|
}
|
|
|
|
const int width = 4 * height() / 3;
|
|
const QRect rect2 = {rect().center().x() - width / 2, rect().top(), width, rect().height()};
|
|
const QRect valid_rect = {is_rhd ? rect2.right() - rect2.height() / 2 : rect2.left(), rect2.top(), rect2.height() / 2, rect2.height()};
|
|
|
|
// blackout
|
|
const QColor bg(0, 0, 0, 140);
|
|
const QRect& blackout_rect = rect();
|
|
p.fillRect(blackout_rect.adjusted(0, 0, valid_rect.left() - blackout_rect.right(), 0), bg);
|
|
p.fillRect(blackout_rect.adjusted(valid_rect.right() - blackout_rect.left(), 0, 0, 0), bg);
|
|
p.fillRect(blackout_rect.adjusted(valid_rect.left()-blackout_rect.left()+1, 0, valid_rect.right()-blackout_rect.right()-1, -valid_rect.height()*7/10), bg); // top dz
|
|
|
|
// face bounding box
|
|
cereal::DriverState::Reader driver_state = sm["driverState"].getDriverState();
|
|
bool face_detected = driver_state.getFaceProb() > 0.5;
|
|
if (face_detected) {
|
|
auto fxy_list = driver_state.getFacePosition();
|
|
auto std_list = driver_state.getFaceOrientationStd();
|
|
float face_x = fxy_list[0];
|
|
float face_y = fxy_list[1];
|
|
float face_std = std::max(std_list[0], std_list[1]);
|
|
|
|
float alpha = 0.7;
|
|
if (face_std > 0.08) {
|
|
alpha = std::max(0.7 - (face_std-0.08)*7, 0.0);
|
|
}
|
|
const int box_size = 0.6 * rect2.height() / 2;
|
|
const float rhd_offset = 0.05; // lhd is shifted, so rhd is not mirrored
|
|
int fbox_x = valid_rect.center().x() + (is_rhd ? (face_x + rhd_offset) : -face_x) * valid_rect.width();
|
|
int fbox_y = valid_rect.center().y() + face_y * valid_rect.height();
|
|
p.setPen(QPen(QColor(255, 255, 255, alpha * 255), 10));
|
|
p.drawRoundedRect(fbox_x - box_size / 2, fbox_y - box_size / 2, box_size, box_size, 35.0, 35.0);
|
|
}
|
|
|
|
// icon
|
|
const int img_offset = 30;
|
|
const int img_x = is_rhd ? rect2.right() - FACE_IMG_SIZE - img_offset : rect2.left() + img_offset;
|
|
const int img_y = rect2.bottom() - FACE_IMG_SIZE - img_offset;
|
|
p.setOpacity(face_detected ? 1.0 : 0.3);
|
|
p.drawPixmap(img_x, img_y, face_img);
|
|
}
|
|
|