你还没有说你希望窗口遵循什么样的路径。如果路径是一些简单的时间函数——也就是说,如果你有办法计算你希望窗口在任何给定时间的位置——你可以尝试下面代码中说明的方法。对于示例中非常简单的菜单,它在我的 Linux 系统上运行良好,并且产生了相当流畅的动作。
该方法的关键在于,它不是在每个计时器事件中将窗口移动给定距离,而是找出当前时间并将窗口移动到当时应该在的位置。因此,运动速度的时间导数应该是恒定的,即使定时器事件不规则地发生,这也可以避免不规则或不稳定的运动。 (如g-timeout-add() description 中所述,很容易出现异常。)
在这个例子中,路径是从窗口的左上角到左下角然后返回,重复。 timerEvent() 中的常量“HalfTime”控制从一个角落移动到另一个角落所需的时间。 g_timeout_add() 调用中的常量 3 将计时器间隔设置为 0.003 秒,或大约每秒 333 次移动 (MPS)。 (您可能想尝试更合理的速率,例如 20、30、40 等 MPS;我使用数字 3 是因为我在使用之前没有查看g-timeout-add(),并假设延迟为数百秒,大约 33 MPS,而不是毫秒,大约 333 MPS。)如果您的窗口内容非常复杂,则可以使用更少的 MPS。此外,我尝试了一些较慢的速度,并得到了更多的波涛汹涌的印象。
/* $Id: app12.c $
Re: animating position of a top-level Gtk window
jiw July 2011 -- Offered without warranty under GPL v3
terms per http://www.gnu.org/licenses/gpl.html */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <gtk/gtk.h>
typedef struct DATA { GTimer *timer; GtkWidget *window; int w, h; }
DataStruct;
gboolean timerEvent(void *dataset) {
enum { HalfTime=8, CycTime=2*HalfTime };
gulong micros;
DataStruct *data =dataset;
double t = fabs(fmod (g_timer_elapsed (data->timer, µs), CycTime));
int x = (t*data->w)/HalfTime, y = (t*data->h)/HalfTime;
gtk_window_move (GTK_WINDOW(data->window),
t<HalfTime? x : 2*data->w-x, t<HalfTime? y : 2*data->h-y);
return TRUE; /* Keep timeout running */
}
int main(int argc, char **argv) {
GtkWidget *vbox, *b;
GdkScreen *gds;
DataStruct data;
data.timer = g_timer_new();
gtk_init (&argc, &argv);
data.window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_default_size (GTK_WINDOW(data.window), 200, 150);
g_signal_connect (G_OBJECT(data.window), "destroy",
G_CALLBACK(gtk_main_quit), NULL);
vbox = gtk_vbox_new (FALSE, 0);
gtk_container_add (GTK_CONTAINER(data.window), vbox);
b = gtk_button_new_with_label ("Click to Exit");
gtk_box_pack_start (GTK_BOX(vbox), b, TRUE, TRUE, TRUE);
g_signal_connect (b, "clicked", G_CALLBACK(gtk_main_quit), NULL);
gtk_widget_show_all (data.window);
gds = gdk_screen_get_default (); /* Get pointer to screen */
data.w = gdk_screen_get_width (gds); /* Find out screen width */
data.h = gdk_screen_get_height (gds); /* Find out screen height */
printf ("Screen size = %d by %d\n", data.w, data.h); fflush(stdout);
g_timeout_add(3, timerEvent, &data); /* Create .003 sec timer */
gtk_main();
return (0);
}