1

我是 Cairo 和 GTK 的新手,我正在开发的程序需要绘制一个 500x500 或 1000x1000 的圆形平铺。此外,在绘图之前还有一些工作要做,但现在我专注于绘图部分,这将涉及到鼠标交互来改变任何圆圈的颜色。

所以,平铺是相同的,随着时间的推移,圆圈必须改变它们的颜色(全部)。我必须检查每个圆圈并执行操作,并且在检查所有圆圈之后,我必须显示更改。此过程必须执行任意次数。

现在我有一个带有滚动窗口的平铺,但是仅仅这样滚动就需要很多时间。提前致谢。我的代码是下一个:

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

static void do_drawing(cairo_t *, GtkWidget *);
static int cellRadius=5;
static int cellDiameter=10;
static int latticeSideSize=500;

static gboolean on_draw_event(GtkWidget *widget, cairo_t *cr, gpointer user_data){  
    do_drawing(cr, widget);  
    return FALSE;
}

static void do_drawing(cairo_t *cr, GtkWidget *widget)
{
  int i=0,j=0;
  GtkWidget *win = gtk_widget_get_toplevel(widget);

  int width, height;
  gtk_window_get_size(GTK_WINDOW(win), &width, &height);

  cairo_set_line_width(cr, .5);  
  cairo_set_source_rgb(cr, 0.69, 0.19, 0);
  cairo_save (cr);

  for(i=0;i<latticeSideSize;i++){

    for(j=0;j<latticeSideSize;j++){
      if(i%2 == 0){
        cairo_arc(cr, cellRadius + 2*cellRadius + j*cellDiameter, cellRadius + cellRadius + i*cellDiameter, cellRadius, 0, 2 * M_PI);
        cairo_stroke(cr);
      }else{
        cairo_arc(cr, cellRadius + cellRadius + j*cellDiameter, cellRadius + cellRadius + i*cellDiameter, cellRadius, 0, 2 * M_PI);
        cairo_stroke(cr);
      }

    }

  }
  cairo_restore (cr);
}

static void destroy( GtkWidget *widget, gpointer   data ){
    gtk_main_quit ();
}

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

  gtk_init(&argc, &argv);

  window = gtk_window_new(GTK_WINDOW_TOPLEVEL); 
  scrolled_window = gtk_scrolled_window_new (NULL, NULL); 
  darea = gtk_drawing_area_new();
  gtk_container_add(GTK_CONTAINER(scrolled_window), darea);
  gtk_container_add(GTK_CONTAINER(window), scrolled_window);

  g_signal_connect(G_OBJECT(darea), "draw", G_CALLBACK(on_draw_event), NULL);
  g_signal_connect(G_OBJECT(scrolled_window), "destroy", G_CALLBACK(gtk_main_quit), NULL);

  gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  gtk_widget_set_size_request( scrolled_window, 500, 500 );
  gtk_window_set_default_size(GTK_WINDOW(window), 1024, 800); 
  gtk_widget_set_hexpand( scrolled_window, TRUE );
  gtk_widget_set_vexpand( scrolled_window, TRUE );
  gtk_window_set_title(GTK_WINDOW(window), "HexaGrid");
  gtk_widget_set_size_request(darea,cellDiameter*latticeSideSize + 20,cellDiameter*latticeSideSize + 20);

  gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
  gtk_container_set_border_width(GTK_CONTAINER (scrolled_window), 10);

  gtk_widget_show_all(window);

  gtk_main();

  return 0;
}
4

1 回答 1

4

这里有两个问题。首先,每次绘制都要进行 250000 次相同的相当苛刻的计算(圆圈),因此如果滚动流畅,则每秒 1500 万圈:这不是一个现实的要求。您可能应该做一次圆,然后使用 CAIRO_EXTEND_REPEAT 扩展模式应用与表面图案相同的结果。您可以通过使用设置图案的位置,cairo_translate()并使用cairo_set_source()将您的圆形图案设置为源,然后使用cairo_rectangle()+cairo_fill()来绘制它。Cairo 示例包含一个使用位图的示例。

如果某些圆圈需要使用不同的颜色,您可以“手动”绘制部分或全部(没有重复扩展模式),但使用图案仍然可能是避免多次计算圆圈的好主意。

其次,在复杂小部件的情况下,不绘制整个小部件而只绘制脏区域是有意义的:请参阅绘图信号文档

于 2014-03-16T11:36:49.900 回答