소스 검색

Fix stuttering animations when using high frame rate limit or vsync

Alexander Wilms 2 년 전
부모
커밋
e2eeec96a9
1개의 변경된 파일18개의 추가작업 그리고 9개의 파일을 삭제
  1. 18 9
      client/gui/FramerateManager.cpp

+ 18 - 9
client/gui/FramerateManager.cpp

@@ -12,27 +12,36 @@
 #include "FramerateManager.h"
 
 #include "../../lib/CConfigHandler.h"
+#include <SDL_video.h>
 
 FramerateManager::FramerateManager(int targetFrameRate)
-	: targetFrameTime(Duration(boost::chrono::seconds(1)) / targetFrameRate)
-	, lastFrameIndex(0)
+	: lastFrameIndex(0)
 	, lastFrameTimes({})
 	, lastTimePoint(Clock::now())
 	, vsyncEnabled(settings["video"]["vsync"].Bool())
 {
+	if(vsyncEnabled)
+	{
+		static int display_in_use = 0;
+		SDL_DisplayMode mode;
+		SDL_GetCurrentDisplayMode(display_in_use, &mode);
+		int displayRefreshRate = mode.refresh_rate;
+		logGlobal->info("Display refresh rate is %d", displayRefreshRate);
+		targetFrameTime = Duration(boost::chrono::seconds(1)) / displayRefreshRate;
+	} else
+	{
+		targetFrameTime = Duration(boost::chrono::seconds(1)) / targetFrameRate;
+	}
 	boost::range::fill(lastFrameTimes, targetFrameTime);
 }
 
 void FramerateManager::framerateDelay()
 {
-	if(!vsyncEnabled)
-	{
-		Duration timeSpentBusy = Clock::now() - lastTimePoint;
+	Duration timeSpentBusy = Clock::now() - lastTimePoint;
 
-		// FPS is higher than it should be, then wait some time
-		if(timeSpentBusy < targetFrameTime)
-			boost::this_thread::sleep_for(targetFrameTime - timeSpentBusy);
-	}
+	// FPS is higher than it should be, then wait some time
+	if(timeSpentBusy < targetFrameTime)
+		boost::this_thread::sleep_for(targetFrameTime - timeSpentBusy);
 
 	// compute actual timeElapsed taking into account actual sleep interval
 	// limit it to 100 ms to avoid breaking animation in case of huge lag (e.g. triggered breakpoint)