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.
		
		
		
		
		
			
		
			
				
					
					
						
							212 lines
						
					
					
						
							7.2 KiB
						
					
					
				
			
		
		
	
	
							212 lines
						
					
					
						
							7.2 KiB
						
					
					
				| #include "tools/cabana/videowidget.h"
 | |
| 
 | |
| #include <QBuffer>
 | |
| #include <QButtonGroup>
 | |
| #include <QDateTime>
 | |
| #include <QHBoxLayout>
 | |
| #include <QMouseEvent>
 | |
| #include <QPainter>
 | |
| #include <QPixmap>
 | |
| #include <QStyleOptionSlider>
 | |
| #include <QTimer>
 | |
| #include <QToolTip>
 | |
| #include <QVBoxLayout>
 | |
| #include <QtConcurrent>
 | |
| 
 | |
| inline QString formatTime(int seconds) {
 | |
|   return QDateTime::fromTime_t(seconds).toString(seconds > 60 * 60 ? "hh:mm:ss" : "mm:ss");
 | |
| }
 | |
| 
 | |
| VideoWidget::VideoWidget(QWidget *parent) : QFrame(parent) {
 | |
|   setFrameShape(QFrame::StyledPanel);
 | |
|   setFrameShadow(QFrame::Sunken);
 | |
|   QHBoxLayout *containter_layout = new QHBoxLayout(this);
 | |
|   QVBoxLayout *main_layout = new QVBoxLayout();
 | |
|   main_layout->setContentsMargins(0, 0, 0, 0);
 | |
| 
 | |
|   containter_layout->addStretch(1);
 | |
|   containter_layout->addLayout(main_layout);
 | |
|   containter_layout->addStretch(1);
 | |
| 
 | |
|   cam_widget = new CameraWidget("camerad", can->visionStreamType(), false, this);
 | |
|   cam_widget->setFixedSize(parent->width(), parent->width() / 1.596);
 | |
|   main_layout->addWidget(cam_widget);
 | |
| 
 | |
|   // slider controls
 | |
|   QHBoxLayout *slider_layout = new QHBoxLayout();
 | |
|   QLabel *time_label = new QLabel("00:00");
 | |
|   slider_layout->addWidget(time_label);
 | |
| 
 | |
|   slider = new Slider(this);
 | |
|   slider->setSingleStep(0);
 | |
|   slider_layout->addWidget(slider);
 | |
| 
 | |
|   end_time_label = new QLabel(this);
 | |
|   slider_layout->addWidget(end_time_label);
 | |
|   main_layout->addLayout(slider_layout);
 | |
| 
 | |
|   // btn controls
 | |
|   QHBoxLayout *control_layout = new QHBoxLayout();
 | |
|   play_btn = new QPushButton("⏸");
 | |
|   play_btn->setStyleSheet("font-weight:bold; height:16px");
 | |
|   control_layout->addWidget(play_btn);
 | |
| 
 | |
|   QButtonGroup *group = new QButtonGroup(this);
 | |
|   group->setExclusive(true);
 | |
|   for (float speed : {0.1, 0.5, 1., 2.}) {
 | |
|     QPushButton *btn = new QPushButton(QString("%1x").arg(speed), this);
 | |
|     btn->setCheckable(true);
 | |
|     QObject::connect(btn, &QPushButton::clicked, [=]() { can->setSpeed(speed); });
 | |
|     control_layout->addWidget(btn);
 | |
|     group->addButton(btn);
 | |
|     if (speed == 1.0) btn->setChecked(true);
 | |
|   }
 | |
|   main_layout->addLayout(control_layout);
 | |
| 
 | |
|   QObject::connect(slider, &QSlider::sliderReleased, [this]() { can->seekTo(slider->value() / 1000.0); });
 | |
|   QObject::connect(slider, &QSlider::valueChanged, [=](int value) { time_label->setText(formatTime(value / 1000)); });
 | |
|   QObject::connect(cam_widget, &CameraWidget::clicked, []() { can->pause(!can->isPaused()); });
 | |
|   QObject::connect(play_btn, &QPushButton::clicked, []() { can->pause(!can->isPaused()); });
 | |
|   QObject::connect(can, &CANMessages::updated, this, &VideoWidget::updateState);
 | |
|   QObject::connect(can, &CANMessages::paused, [this]() { play_btn->setText("▶"); });
 | |
|   QObject::connect(can, &CANMessages::resume, [this]() { play_btn->setText("⏸"); });
 | |
|   QObject::connect(can, &CANMessages::streamStarted, [this]() {
 | |
|     end_time_label->setText(formatTime(can->totalSeconds()));
 | |
|     slider->setRange(0, can->totalSeconds() * 1000);
 | |
|   });
 | |
| }
 | |
| 
 | |
| void VideoWidget::rangeChanged(double min, double max, bool is_zoomed) {
 | |
|   if (!is_zoomed) {
 | |
|     min = 0;
 | |
|     max = can->totalSeconds();
 | |
|   }
 | |
|   end_time_label->setText(formatTime(max));
 | |
|   slider->setRange(min * 1000, max * 1000);
 | |
| }
 | |
| 
 | |
| void VideoWidget::updateState() {
 | |
|   if (!slider->isSliderDown())
 | |
|     slider->setValue(can->currentSec() * 1000);
 | |
| }
 | |
| 
 | |
| // Slider
 | |
| Slider::Slider(QWidget *parent) : QSlider(Qt::Horizontal, parent) {
 | |
|   QTimer *timer = new QTimer(this);
 | |
|   timer->setInterval(2000);
 | |
|   timer->callOnTimeout([this]() {
 | |
|     timeline = can->getTimeline();
 | |
|     update();
 | |
|   });
 | |
|   setMouseTracking(true);
 | |
| 
 | |
|   QObject::connect(can, SIGNAL(streamStarted()), timer, SLOT(start()));
 | |
|   QObject::connect(can, &CANMessages::streamStarted, this, &Slider::streamStarted);
 | |
| }
 | |
| 
 | |
| void Slider::streamStarted() {
 | |
|   abort_load_thumbnail = true;
 | |
|   thumnail_future.waitForFinished();
 | |
|   abort_load_thumbnail = false;
 | |
|   thumbnails.clear();
 | |
|   thumnail_future = QtConcurrent::run(this, &Slider::loadThumbnails);
 | |
| }
 | |
| 
 | |
| void Slider::loadThumbnails() {
 | |
|   const auto segments = can->route()->segments();
 | |
|   for (auto it = segments.rbegin(); it != segments.rend() && !abort_load_thumbnail; ++it) {
 | |
|     std::string qlog = it->second.qlog.toStdString();
 | |
|     if (!qlog.empty()) {
 | |
|       LogReader log;
 | |
|       if (log.load(qlog, &abort_load_thumbnail, {cereal::Event::Which::THUMBNAIL}, true, 0, 3)) {
 | |
|         for (auto ev = log.events.cbegin(); ev != log.events.cend() && !abort_load_thumbnail; ++ev) {
 | |
|           auto thumb = (*ev)->event.getThumbnail();
 | |
|           QString str = getThumbnailString(thumb.getThumbnail());
 | |
|           std::lock_guard lk(thumbnail_lock);
 | |
|           thumbnails[thumb.getTimestampEof()] = str;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| QString Slider::getThumbnailString(const capnp::Data::Reader &data) {
 | |
|   QPixmap thumb;
 | |
|   if (thumb.loadFromData(data.begin(), data.size(), "jpeg")) {
 | |
|     thumb = thumb.scaled({thumb.width()/3, thumb.height()/3}, Qt::KeepAspectRatio);
 | |
|     thumbnail_size = thumb.size();
 | |
|     QByteArray bytes;
 | |
|     QBuffer buffer(&bytes);
 | |
|     buffer.open(QIODevice::WriteOnly);
 | |
|     thumb.save(&buffer, "png");
 | |
|     return  QString("<img src='data:image/png;base64, %0'>").arg(QString(bytes.toBase64()));
 | |
|   }
 | |
|   return {};
 | |
| }
 | |
| 
 | |
| void Slider::sliderChange(QAbstractSlider::SliderChange change) {
 | |
|   if (change == QAbstractSlider::SliderValueChange) {
 | |
|     int x = width() * ((value() - minimum()) / double(maximum() - minimum()));
 | |
|     if (x != slider_x) {
 | |
|       slider_x = x;
 | |
|       update();
 | |
|     }
 | |
|   } else {
 | |
|     QAbstractSlider::sliderChange(change);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void Slider::paintEvent(QPaintEvent *ev) {
 | |
|   static const QColor colors[] = {
 | |
|     [(int)TimelineType::None] = QColor(111, 143, 175),
 | |
|     [(int)TimelineType::Engaged] = QColor(0, 163, 108),
 | |
|     [(int)TimelineType::UserFlag] = Qt::white,
 | |
|     [(int)TimelineType::AlertInfo] = Qt::green,
 | |
|     [(int)TimelineType::AlertWarning] = QColor(255, 195, 0),
 | |
|     [(int)TimelineType::AlertCritical] = QColor(199, 0, 57)};
 | |
| 
 | |
|   QPainter p(this);
 | |
|   QRect r = rect().adjusted(0, 4, 0, -4);
 | |
|   p.fillRect(r, colors[(int)TimelineType::None]);
 | |
|   double min = minimum() / 1000.0;
 | |
|   double max = maximum() / 1000.0;
 | |
|   for (auto [begin, end, type] : timeline) {
 | |
|     if (begin > max || end < min)
 | |
|       continue;
 | |
|     r.setLeft(((std::max(min, (double)begin) - min) / (max - min)) * width());
 | |
|     r.setRight(((std::min(max, (double)end) - min) / (max - min)) * width());
 | |
|     p.fillRect(r, colors[(int)type]);
 | |
|   }
 | |
| 
 | |
|   QStyleOptionSlider opt;
 | |
|   opt.initFrom(this);
 | |
|   opt.minimum = minimum();
 | |
|   opt.maximum = maximum();
 | |
|   opt.subControls = QStyle::SC_SliderHandle;
 | |
|   opt.sliderPosition = value();
 | |
|   style()->drawComplexControl(QStyle::CC_Slider, &opt, &p);
 | |
| }
 | |
| 
 | |
| void Slider::mousePressEvent(QMouseEvent *e) {
 | |
|   QSlider::mousePressEvent(e);
 | |
|   if (e->button() == Qt::LeftButton && !isSliderDown()) {
 | |
|     int value = minimum() + ((maximum() - minimum()) * e->x()) / width();
 | |
|     setValue(value);
 | |
|     emit sliderReleased();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void Slider::mouseMoveEvent(QMouseEvent *e) {
 | |
|   QString thumb;
 | |
|   {
 | |
|     double seconds = (minimum() + e->pos().x() * ((maximum() - minimum()) / (double)width())) / 1000.0;
 | |
|     std::lock_guard lk(thumbnail_lock);
 | |
|     auto it = thumbnails.lowerBound((seconds + can->routeStartTime()) * 1e9);
 | |
|     if (it != thumbnails.end()) {
 | |
|       thumb = it.value();
 | |
|     }
 | |
|   }
 | |
|   QPoint pt = mapToGlobal({e->pos().x() - thumbnail_size.width() / 2, -thumbnail_size.height() - 30});
 | |
|   QToolTip::showText(pt, thumb, this, rect());
 | |
|   QSlider::mouseMoveEvent(e);
 | |
| }
 | |
| 
 |