Как перерисовать текст с помощью cairo с помощью gtk+

Я использую Каир, чтобы нарисовать такой круговой виджет, как этот.

Я использовал событие "рисовать" области рисования, чтобы нарисовать весь виджет, но я не знаю, как обновить данные в виджете. Это код, который я делаю:

gboolean hald_circular_gauge (GtkWidget *widget,cairo_t *cr, gdouble max, gdouble min, gdouble value, gchar *pcname ,gchar *unit){

cairo_text_extents_t extents, extentsTemp, extentsLabel, extentsTempMin, extentsTempMax;
  int width, height;
  gint percentage, linewidth;
  double x,y;
  gdouble declive = (360.0-180.0)/(max-min);
  gdouble origin = 180.0 - declive*min;
  gchar pcTemp[BUFSIZ];

  width = gtk_widget_get_allocated_width (widget);
  height = gtk_widget_get_allocated_height (widget);



  double angle1 = 180.0  * (M_PI/180.0);  /* angles are specified */
  double angle2 = 360.0 * (M_PI/180.0);  /* in radians           */

  if(min > value || value >max)
      return FALSE;

  linewidth = (MIN (width, height) / 2.0 * 30.0) / 100.0;

  cairo_set_source_rgba (cr, 1.0, 1.0, 1.0, 0.6);
  cairo_set_line_width (cr, linewidth);
  cairo_arc(cr, width/2.0, height/2.0,  MIN (width, height) / 2.0 - linewidth, angle1, angle2);
  cairo_stroke (cr);

  cairo_set_source_rgba (cr, 0.0, 0.9, 0.0, 1.0);
  cairo_set_line_width (cr, linewidth);
  cairo_arc(cr, width/2.0, height/2.0, MIN (width, height) / 2.0  - linewidth, 180.0  * (M_PI/180.0),(declive*value+origin)  * (M_PI/180.0) );
  cairo_stroke (cr);

  cairo_select_font_face (cr, "Roboto Thin", CAIRO_FONT_SLANT_NORMAL,
          CAIRO_FONT_WEIGHT_NORMAL);
  cairo_set_font_size (cr, linewidth);
  cairo_set_source_rgb (cr, 1.0, 1.0, 1);
  cairo_text_extents (cr, unit, &extents);
  x = width/2.0-(extents.width/2 + extents.x_bearing);
  y = height/2.0 - (extents.height/2 );
  cairo_move_to (cr, x, y);
  cairo_show_text (cr, unit);

  cairo_select_font_face (cr, "Roboto", CAIRO_FONT_SLANT_NORMAL,
              CAIRO_FONT_WEIGHT_BOLD);
  cairo_set_font_size (cr, linewidth*1.3);
  sprintf(pcTemp,"%.1f",value);
  cairo_text_extents (cr, pcTemp, &extentsTemp);
      x = width/2.0-(extentsTemp.width/2 + extentsTemp.x_bearing);
      y = height/2.0 - extentsTemp.y_bearing;
  cairo_move_to (cr, x, y);
  cairo_show_text (cr, pcTemp);

  cairo_select_font_face (cr, "Roboto Thin", CAIRO_FONT_SLANT_NORMAL,
                  CAIRO_FONT_WEIGHT_NORMAL);
  cairo_set_font_size (cr, linewidth);
  cairo_text_extents (cr, pcname, &extentsLabel);
      x = width/2.0-(extentsLabel.width/2 + extentsLabel.x_bearing);
      y = height/2.0 - extentsTemp.y_bearing  - extentsLabel.y_bearing*1.20;
  cairo_move_to (cr, x, y);
  cairo_show_text (cr, pcname);


  cairo_select_font_face (cr, "Roboto Thin", CAIRO_FONT_SLANT_NORMAL,
                      CAIRO_FONT_WEIGHT_NORMAL);
  cairo_set_font_size (cr, linewidth*0.5);
  sprintf(pcTemp,"%.1f",min);
  cairo_text_extents (cr, pcTemp, &extentsTempMin);
      x = width/2.0  - MIN (width, height) / 2.0 + linewidth - (extentsTempMin.width/2 + extentsTempMin.x_bearing);
      y = height/2.0 - extentsTempMin.y_bearing*1.20;
  cairo_move_to (cr, x, y);
  cairo_show_text (cr, pcTemp);


  cairo_select_font_face (cr, "Roboto Thin", CAIRO_FONT_SLANT_NORMAL,
                      CAIRO_FONT_WEIGHT_NORMAL);
      cairo_set_font_size (cr, linewidth*0.5);
      sprintf(pcTemp,"%.1f",max);
      cairo_text_extents (cr, pcTemp, &extentsTempMax);
          x = width/2.0  + MIN (width, height) / 2.0 - linewidth + (-extentsTempMax.width/2 + extentsTempMax.x_bearing);
          y = height/2.0 - extentsTempMax.y_bearing*1.20;
      cairo_move_to (cr, x, y);
      cairo_show_text (cr, pcTemp);
    return TRUE;
 }

и результат такой

 GtkWidget *drawing_area = gtk_drawing_area_new ();
     gtk_widget_set_size_request (drawing_area, 100, 100);
     gtk_box_pack_start (GTK_BOX(gtk_builder_get_object(builder, "box30")),drawing_area,FALSE,TRUE,0);

Я создаю область рисования и запрашиваю размер, но я не получаю этот размер. Итак, мой вопрос:

  • Как я могу обновить данные в области рисования (создать событие рисования??)?

  • Почему я не могу запросить размер виджета области рисования?

Мнения приветствуются.

1 ответ

Решение

Ответ на

1) Каждый раз, когда вы обновляете данные, запускайте обновление с помощью:

gtk_widget_queue_draw(GTK_WIDGET(drawing_area))

2) Вы можете запросить размер, но если контейнер настроен на расширение, то виджет может в конечном итоге расширяться; Функция / метод set_size_request установит минимальный размер, который будет иметь область рисования. Из того, что мы видим в вашем коде, область рисования находится внутри GtkBox. Проверьте разверните и заполните флаги.

Совет 1: Вы можете протестировать set_size_request, используя Drawing_area внутри контейнера GtkFixed

Совет 2: попробуйте создать собственный виджет, это было бы лучшим решением.

Другие вопросы по тегам