出于测试目的,我真的希望能够手动控制GTK3应用程序的帧刷新率。变慢很容易,我只是睡在渲染功能中。但是,我可以达到的最大帧速率是我的显示器刷新率为60Hz。
我发现我可以使用gdk_threads_add_idle
在主循环中添加一个无上限的更新槽,但是即使我从这个函数内部调用gtk_gl_area_queue_render
,实际的渲染函数也只能在我的监视器上运行'刷新率。
有没有办法迫使重新渲染成为"立即"根据要求?我尝试过手动调用渲染槽,这确实会导致立即调用重新渲染功能,但它根本不会更新屏幕。
void update_auto(gpointer user_data)
{
// ... do some update stuff
// somehow get the appropriate GtkGLArea from user_data
GtkGLArea* area = get_area(user_data);
gtk_gl_area_queue_render(area);
// capped at 60Hz
}
void update_manual(gpointer user_data)
{
// ... do some update stuff
// somehow get the appropriate GtkGLArea from user_data
GtkGLArea* area = get_area(user_data);
gtk_gl_area_make_current(area);
render(area,gtk_gl_area_get_context(area), user_data);
// even though render was called, screen never updates
}
void render(GtkGLArea* area, GdkGLContext* context, gpointer user_data)
{
// ... render
}
void setup_widget()
{
// ... some init stuff
// only one of these is used at a time
#if TEST_FLAG == 0
gdk_threads_add_idle(update_manual, user_data);
#elif TEST_FLAG == 1
gdk_threads_add_idle(update_auto, user_data);
#elif TEST_FLAG == 2
gtk_widget_add_tick_callback((GtkWidget*) area, update_auto, user_data, NULL);
#else
gtk_widget_add_tick_callback((GtkWidget*) area, update_manual, user_data, NULL);
#endif
}
答案 0 :(得分:0)
我发现了一个强迫立即重画的黑客攻击:
GtkGLArea
扩展了GtkWidget
类,因此我可以使用draw函数立即重新绘制。
// inside update
// get the proper clipping so we don't draw over other widgets
GtkAllocation reg;
gtk_widget_get_allocation((GtkWidget*)area, ®);
cairo_region_t *creg = cairo_region_create_rectangle(®);
cairo_t *cr = gdk_cairo_create(gtk_widget_get_window((GtkWidget*)area));
gdk_cairo_region(cr, creg);
cairo_clip(cr);
gtk_widget_draw((GtkWidget*)area, cr);
cairo_destroy(cr);
cairo_region_destroy(creg);
如果需要,我可以使用gdk_threads_add_idle
函数,它会尽快调用更新函数。
即使不使用gdk_threads_add_idle
,我也注意到即使对gtk_widget_add_tick_callback
使用此方法,帧速率也会更加一致。在我注意到帧速率偶尔在15/30/60 fps之间跳跃之前,似乎表明它丢失了一个或两个重绘更新周期。这个问题现在已经完全消失了,我得到一个接近常数的N fps,其中对于小场景,N的上限为60。
我还没有遇到任何问题但这样做,但这似乎适用于包含多个GtkWidget
的窗口。我没有尝试过重叠的小部件,但我怀疑这里唯一的解决方案是重绘任何重叠的小部件。