【问题标题】:How to rotate image using GTK+ / Cairo如何使用 GTK+ / Cairo 旋转图像
【发布时间】:2012-06-14 17:08:52
【问题描述】:

我有一个简单的应用程序,它应该使用 GTK+Cairo 每 x 毫秒将装饰轮旋转这么多度。我在下面有一些代码可以从计时器调用cairo_rotate()。但是,图像不会改变。我是否必须使图像无效才能触发暴露事件?我对 Cairo 很陌生,我非常感谢一个简单的示例演示如何在 GTK+ 中使用 Cairo 旋转图像。

#include <cairo.h>
#include <gtk/gtk.h>

cairo_surface_t *image;
cairo_t *cr;


gboolean rotate_cb( void )
{

    cairo_rotate (cr, 1);
    //cairo_paint(cr);
    printf("rotating\n");

    return( TRUE );
}

static gboolean
on_expose_event(GtkWidget *widget,
    GdkEventExpose *event,
    gpointer data)
{
  cr = gdk_cairo_create (widget->window);

  cairo_set_source_surface(cr, image, 0, 0);
  cairo_paint(cr);
  printf("Paint\n");

  //cairo_destroy(cr);

  return FALSE;
}


int main(int argc, char *argv[])
{
  GtkWidget *window;

  image = cairo_image_surface_create_from_png("wheel.png");

  gtk_init(&argc, &argv);

  window = gtk_window_new(GTK_WINDOW_TOPLEVEL);

  g_signal_connect(window, "expose-event",
      G_CALLBACK (on_expose_event), NULL);
  g_signal_connect(window, "destroy",
      G_CALLBACK (gtk_main_quit), NULL);

  gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  gtk_window_set_default_size(GTK_WINDOW(window), 500, 500); 
  gtk_widget_set_app_paintable(window, TRUE);

  gtk_widget_show_all(window);
  g_timeout_add(500, (GSourceFunc) rotate_cb, NULL);

  gtk_main();
  cairo_destroy(cr);
  cairo_surface_destroy(image);

  return 0;
}

【问题讨论】:

  • 要意识到的重要一点是 Cairo 不会创建持久对象,因此在您绘制完之后旋转它不会做任何事情。这是 Cairo 和像 GooCanvas 这样的画布库的区别。

标签: c gtk cairo


【解决方案1】:

您需要将旋转存储在一个变量中并将 cairo_rotate(cr, rotation_amt);在绘制之前调用 on_expose_event 方法。

如果图像居中,还可以平移到窗口的中心、旋转并向后平移,以使轮子围绕其中心旋转。

cairo_translate(cr, width / 2.0, height / 2.0);
cairo_rotate(cr, rotation_amt);
cairo_translate(cr, - image_w / 2.0, - image_h / 2.0);
cairo_set_source_surface(cr, image, 0, 0);
cairo_paint(cr);

我希望这是对的。

正如 ptomato 所说,您需要通过从 rotate_cb 调用 gtk_widget_queue_draw 来使绘图表面无效。为 Cairo 上下文保留一个全局变量是多余的。图像不会旋转,因为新创建的上下文加载了单位矩阵,并且您之前的所有转换都被重置。

【讨论】:

  • 您还必须gtk_widget_queue_draw() 在计时器触发时触发公开事件。
【解决方案2】:

在 cairo 邮件列表的帮助下,这是一个工作示例。我将其分享给那些可能觉得这很有用的人。

#include <cairo.h>
#include <gtk/gtk.h>
#include <math.h>
#include <stdlib.h> 

cairo_surface_t *image;
cairo_t *cr;
gdouble rotation = 0;
GtkWidget *window;

gint image_w, image_h;
double DegreesToRadians( int degrees );


double DegreesToRadians( int degrees )
{
    return((double)((double)degrees * ( M_PI/180 )));
}


gboolean rotate_cb( void *degrees )
{
    // Any rotation applied to cr here will be lost, as we create
    // a new cairo context on every expose event
    //cairo_rotate (cr, 4);
    rotation += DegreesToRadians((*(int*)(degrees)));
    //cairo_paint(cr);
    //      printf("rotating\n");
    // Tell our window that it should repaint itself (ie. emit an expose event)
    gtk_widget_queue_draw(window);

    return( TRUE );
}

static gboolean on_expose_event(GtkWidget *widget, GdkEventExpose *event,gpointer data)
{
    // Make sure our window wasn't destroyed yet
    // (to silence a warning)
    g_return_if_fail(GTK_IS_WIDGET(widget));

    cr = gdk_cairo_create (widget->window);
    // We need to apply transformation before setting the source surface
    // We translate (0, 0) to the center of the screen,
    // so we can rotate the image around its center point,
    // not its upper left corner
    cairo_translate(cr, image_w/2, image_h/2);
    cairo_rotate(cr, rotation);
    cairo_set_source_surface(cr, image, -image_w/2, -image_h/2);
    // We need to clip around the image, or cairo will paint garbage data
    //cairo_rectangle(cr, -image_w/2, -image_h/2, image_w, image_h);
    //cairo_clip(cr);

    cairo_paint(cr);
    //printf("Paint\n");
    cairo_destroy(cr);
    return FALSE;
}


int main(int argc, char *argv[])
{   
    int degrees = 10, speed = 125;
    image = cairo_image_surface_create_from_png("wheel.png");
    image_w = cairo_image_surface_get_width(image);
    image_h = cairo_image_surface_get_height(image);

    gtk_init(&argc, &argv);

    if( argc == 3 )
    {
        degrees = atoi(argv[1]);
        speed = atoi(argv[2]);
    }


    window = gtk_window_new(GTK_WINDOW_TOPLEVEL);

    g_signal_connect(window, "expose-event",
    G_CALLBACK (on_expose_event), NULL);
    g_signal_connect(window, "destroy",
    G_CALLBACK (gtk_main_quit), NULL);

    gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
    gtk_window_set_default_size(GTK_WINDOW(window), image_w, image_h);
    gtk_widget_set_app_paintable(window, TRUE);

    gtk_widget_show_all(window);
    g_timeout_add(speed, (GSourceFunc) rotate_cb, (void *)&degrees);

    gtk_main();
    cairo_surface_destroy(image);

    return 0;
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-08-03
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多