事件驱动(事件盒、固定窗口)

最后更新于:2022-04-01 20:22:10

### 事件盒 The EventBox 一 些 GTK 构件没有与之相关联的 X 窗口,所以它们只在其父构件上显示其外观。由于这个原因,它们不能接收任何事件,并且,如果它们尺寸设置不正确,它们也不会自动剪裁(译者注:裁剪就是使 构件只显示一部分),这样可能会把界面弄得乱糟糟的。如果要想构件接收事件,可以使用事件盒(EventBox)。 初 一看,事件盒构件好像完全没有什么用。它在屏幕上什么也不画,并且对事件也不做响应。但是,它有一个功能:为它的子构件提供一个 X 窗口。因为许多G T K构件并没有相关联的 X 窗口,所以这一点很重要。虽然没有 X 窗口会节省内存,提高系统性能,但它也有一些弱点。没有 X 窗口的构件不能接收事件,并且对它的任何内容不能实施剪裁。虽然事件盒构件的名称事件盒强调了它的事件处理功能,它也能用于剪裁构件。看下面的具体代码: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-08-24_57bd7799a2199.jpg) ### 点击将会退出 ~~~ /*File:color.c *Date:2014-01-06 *Author:sjin *Mail:413977243@qq.com */ #include #include #include int main( int argc, char *argv[] ) { /* GtkWidget 是构件的存储类型 */ GtkWidget *window; GtkWidget *event_box; GtkWidget *label; /* 这个函数在所有的 GTK 程序都要调用。参数由命令行中解析出来并且送到该程序中*/ gtk_init (&argc, &argv); /* 创建一个新窗口 */ window = gtk_window_new (GTK_WINDOW_TOPLEVEL); /*设置窗口标题*/ gtk_window_set_title(GTK_WINDOW(window),"My first program helloworld!"); /**/ gtk_window_set_policy(GTK_WINDOW(window),TRUE,TRUE,TRUE); /* 当窗口收到 "delete_event" 信号 (这个信号由窗口管理器发出,通常是“关闭” * 选项或是标题栏上的关闭按钮发出的),我们让它调用在前面定义的 destroy_window() 函数。 * 传给回调函数的 data 参数值是 NULL,它会被回调函数忽略。*/ g_signal_connect (GTK_OBJECT(window), "destroy",G_CALLBACK (gtk_main_quit), (gpointer)window); /*创建一个事件盒,并将它加到顶级窗口上*/ event_box = gtk_event_box_new(); gtk_container_add(GTK_CONTAINER(window),event_box); gtk_widget_show(event_box); /*创建一个长标签*/ label = gtk_label_new("点击这里,退出"); gtk_container_add(GTK_CONTAINER(event_box),label); gtk_widget_show(label); /*将标签剪裁短*/ gtk_widget_set_size_request(label,110,20); /**/ gtk_widget_set_events(event_box,GDK_BUTTON_PRESS_MASK); g_signal_connect(GTK_OBJECT(event_box),"button_press_event",G_CALLBACK(gtk_main_quit),NULL); gtk_widget_realize(event_box); gdk_window_set_cursor(event_box->window,gdk_cursor_new(GDK_HAND1)); gtk_widget_show (window); /* 所有的 GTK 程序必须有一个 gtk_main() 函数。程序运行停在这里 * 等待事件 (如键盘事件或鼠标事件) 的发生。*/ gtk_main (); return 0; } ~~~ ### 固定容器 Fixed Container 固定容器(The Fixed container)允许将构件放在窗口的固定位置,这个位置是相对于固定容器的左上角的。构件的位置可以动态改变。 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-08-24_57bd7799b522a.jpg) ~~~ /*File:color.c *Date:2014-01-07 *Author:sjin *Mail:413977243@qq.com */ #include #include #include /*用全局变量储存固定容器里构件的位置*/ gint x = 50; gint y = 50; /*这个回调函数将按钮移动到固定容器的位置*/ void move_button(GtkWidget *widget,GtkWidget *fixed) { x = (x + 30)%300; y = (y + 30)%300; gtk_fixed_move(GTK_FIXED(fixed),widget,x,y); } int main( int argc, char *argv[] ) { /* GtkWidget 是构件的存储类型 */ GtkWidget *window; GtkWidget *button; GtkWidget *fixed; gint i; /* 这个函数在所有的 GTK 程序都要调用。参数由命令行中解析出来并且送到该程序中*/ gtk_init (&argc, &argv); /* 创建一个新窗口 */ window = gtk_window_new (GTK_WINDOW_TOPLEVEL); /*设置窗口标题*/ gtk_window_set_title(GTK_WINDOW(window),"Fixed Container"); /**/ gtk_window_set_policy(GTK_WINDOW(window),TRUE,TRUE,TRUE); /* 当窗口收到 "delete_event" 信号 (这个信号由窗口管理器发出,通常是“关闭” * 选项或是标题栏上的关闭按钮发出的),我们让它调用在前面定义的 destroy_window() 函数。 * 传给回调函数的 data 参数值是 NULL,它会被回调函数忽略。*/ g_signal_connect (GTK_OBJECT(window), "destroy",G_CALLBACK (gtk_main_quit), NULL); /*设置窗口的边框宽度*/ gtk_container_set_border_width(GTK_CONTAINER(window),10); /*创建一个固定容器*/ fixed = gtk_fixed_new(); gtk_container_add(GTK_CONTAINER(window),fixed); gtk_widget_show(fixed); for(i = 0; i < 3; i++){ /*创建一个按钮*/ button = gtk_button_new_with_label("miss you!"); g_signal_connect(GTK_OBJECT(button),"clicked",G_CALLBACK(move_button),fixed); /*将按钮组装到一个固定容器的窗口中*/ gtk_fixed_put(GTK_FIXED(fixed),button,i*50,i*50); gtk_widget_show(button); } gtk_widget_show (window); /* 所有的 GTK 程序必须有一个 gtk_main() 函数。程序运行停在这里 * 等待事件 (如键盘事件或鼠标事件) 的发生。*/ gtk_main (); return 0; } ~~~
';