|  |  | @ -2,6 +2,7 @@ | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <QApplication> |  |  |  | #include <QApplication> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <QTranslator> |  |  |  | #include <QTranslator> | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | #include <QWindow> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <selfdrive/ui/qt/util.h> |  |  |  | #include <selfdrive/ui/qt/util.h> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <selfdrive/ui/qt/window.h> |  |  |  | #include <selfdrive/ui/qt/window.h> | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -73,30 +74,54 @@ Application::Application(int argc, char *argv[], QObject *parent) : QObject(pare | 
			
		
	
		
		
			
				
					
					|  |  |  |   recorder->moveToThread(recorderThread); |  |  |  |   recorder->moveToThread(recorderThread); | 
			
		
	
		
		
			
				
					
					|  |  |  |   connect(recorderThread, &QThread::finished, recorder, &QObject::deleteLater); |  |  |  |   connect(recorderThread, &QThread::finished, recorder, &QObject::deleteLater); | 
			
		
	
		
		
			
				
					
					|  |  |  |   connect(app, &QCoreApplication::aboutToQuit, recorderThread, &QThread::quit); |  |  |  |   connect(app, &QCoreApplication::aboutToQuit, recorderThread, &QThread::quit); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   recorderThread->start(); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   // Initialize and start replay
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   initReplay(route.toStdString()); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   replayThread = QThread::create([this, startTime] { startReplay(startTime); }); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   replayThread->start(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   QTimer *loop = new QTimer; |  |  |  |   // Frame capture optimization
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   connect(loop, &QTimer::timeout, this, [&]() { |  |  |  |   QElapsedTimer frameTimer; | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   frameTimer.start(); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   int64_t lastFrameTime = 0; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   const int64_t frameInterval = 1000 / UI_FREQ;  // Target frame interval in ms
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   loop = new QTimer; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   connect(loop, &QTimer::timeout, this, [&, frameTimer, lastFrameTime]() mutable { | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (!window->isVisible()) { |  |  |  |     if (!window->isVisible()) { | 
			
		
	
		
		
			
				
					
					|  |  |  |       return; |  |  |  |       return; | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |     QElapsedTimer timer; |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     timer.start(); |  |  |  |     int64_t currentTime = frameTimer.elapsed(); | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     int64_t elapsedSinceLastFrame = currentTime - lastFrameTime; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     // Skip frame if we're ahead of schedule
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     if (elapsedSinceLastFrame < frameInterval) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       return; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     QPixmap pixmap = window->grab(); |  |  |  |     QPixmap pixmap = window->grab(); | 
			
		
	
		
		
			
				
					
					|  |  |  |     qDebug() << "pixmap took " << timer.elapsed() << " ms"; |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     timer.restart(); |  |  |  |     // Only process frame if capture was successful
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     recorder->saveFrame(std::make_shared<QPixmap>(std::move(pixmap))); |  |  |  |     if (!pixmap.isNull()) { | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       recorder->saveFrame(std::make_shared<QPixmap>(std::move(pixmap))); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       lastFrameTime = currentTime; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |   }); |  |  |  |   }); | 
			
		
	
		
		
			
				
					
					|  |  |  |   loop->start(1000 / UI_FREQ); |  |  |  | 
 | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   // Use a higher timer resolution for more precise frame timing
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   loop->setTimerType(Qt::PreciseTimer); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   loop->start(1);  // Run at highest possible frequency, we'll control frame rate ourselves
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   window->setAttribute(Qt::WA_DontShowOnScreen); |  |  |  |   window->setAttribute(Qt::WA_DontShowOnScreen); | 
			
		
	
		
		
			
				
					
					|  |  |  |   window->setAttribute(Qt::WA_OpaquePaintEvent); |  |  |  |   window->setAttribute(Qt::WA_OpaquePaintEvent); | 
			
		
	
		
		
			
				
					
					|  |  |  |   window->setAttribute(Qt::WA_NoSystemBackground); |  |  |  |   window->setAttribute(Qt::WA_NoSystemBackground); | 
			
		
	
		
		
			
				
					
					|  |  |  |   recorderThread->start(); |  |  |  |   window->setAttribute(Qt::WA_TranslucentBackground, false); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |   window->setAttribute(Qt::WA_AlwaysStackOnTop); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   // Initialize and start replay
 |  |  |  |   window->setAttribute(Qt::WA_ShowWithoutActivating); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   initReplay(route.toStdString()); |  |  |  |   window->setAttribute(Qt::WA_UpdatesDisabled); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   replayThread = QThread::create([this, startTime] { startReplay(startTime); }); |  |  |  |   window->setAttribute(Qt::WA_StaticContents); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   replayThread->start(); |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | void Application::initReplay(const std::string& route) { |  |  |  | void Application::initReplay(const std::string& route) { | 
			
		
	
	
		
		
			
				
					|  |  | @ -132,15 +157,18 @@ Application::~Application() { | 
			
		
	
		
		
			
				
					
					|  |  |  |   if (recorderThread) { |  |  |  |   if (recorderThread) { | 
			
		
	
		
		
			
				
					
					|  |  |  |     recorderThread->quit(); |  |  |  |     recorderThread->quit(); | 
			
		
	
		
		
			
				
					
					|  |  |  |     recorderThread->wait(); |  |  |  |     recorderThread->wait(); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     delete recorderThread; | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   delete recorder; | 
			
		
	
		
		
			
				
					
					|  |  |  |   delete window; |  |  |  |   delete window; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   delete loop; | 
			
		
	
		
		
			
				
					
					|  |  |  |   delete app; |  |  |  |   delete app; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | int Application::exec() const { |  |  |  | int Application::exec() const { | 
			
		
	
		
		
			
				
					
					|  |  |  |   // TODO: modify Replay to block until all OnroadWindow required messages have been broadcast at least once
 |  |  |  |   // TODO: modify Replay to block until all OnroadWindow required messages have been broadcast at least once
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   std::this_thread::sleep_for(std::chrono::seconds(5)); |  |  |  |   std::this_thread::sleep_for(std::chrono::seconds(8)); | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |   setMainWindow(window); |  |  |  |   setMainWindow(window); | 
			
		
	
		
		
			
				
					
					|  |  |  |   return app->exec(); |  |  |  |   return app->exec(); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
	
		
		
			
				
					|  |  | 
 |