QQ登录

只需一步,快速开始

 找回密码
 注册

QQ登录

只需一步,快速开始

查看: 5320|回复: 27

gtk图像处理问题

[复制链接]
发表于 2005-8-3 12:48:39 | 显示全部楼层 |阅读模式
在回调函数中改变了图像的像素值,如何刷新显示图像呢
还是应该刷新窗口的显示呢
发表于 2005-8-3 13:03:51 | 显示全部楼层
背景图还是贴上去的。
回复

使用道具 举报

 楼主| 发表于 2005-8-3 13:45:41 | 显示全部楼层
是用如下几句显示的
GtkWidget *image;
image= gtk_image_new_from_file ("x.bmp");
gtk_widget_show(image);

然后在回调函数中改变了图像的像素值,我重新用
gtk_widget_show(image);
gtk_widget_show(window);
都不能刷新图像,应该怎么做呢?
3x
回复

使用道具 举报

发表于 2005-8-3 14:06:32 | 显示全部楼层
执行gdk_flush();试试
回复

使用道具 举报

 楼主| 发表于 2005-8-3 14:08:59 | 显示全部楼层
好的,等会试试
晚上看结果
谢谢版主
回复

使用道具 举报

 楼主| 发表于 2005-8-3 20:03:41 | 显示全部楼层
唉呀,好象还是不行啊,还有其他的办法吗?
回复

使用道具 举报

发表于 2005-8-3 20:26:35 | 显示全部楼层
gtk_widget_draw
回复

使用道具 举报

 楼主| 发表于 2005-8-3 20:44:14 | 显示全部楼层
good,正是我想要的,太感谢了         
回复

使用道具 举报

发表于 2005-8-4 08:43:18 | 显示全部楼层
void        gtk_widget_queue_draw_area      (GtkWidget *widget,
                                             gint x,
                                             gint y,
                                             gint width,
                                             gint height);
但是改变像素值,会改变图像的样子吗?
回复

使用道具 举报

发表于 2005-8-4 10:03:43 | 显示全部楼层
然后在回调函数中改变了图像的像素值,我重新用
gtk_widget_show(image);
gtk_widget_show(window);
都不能刷新图像,应该怎么做呢?
3x


你既然是 new_from_file, 如何改的像素值?? 建议试试使用 new_from_pixbuf/bitmap 然后 改了 pixbuf/bitmap, 然后再 set_from_pixbuf/bitmap
回复

使用道具 举报

 楼主| 发表于 2005-8-5 03:28:16 | 显示全部楼层
[quote:9274450700="Kan"]void        gtk_widget_queue_draw_area      (GtkWidget *widget,
                                             gint x,
                                             gint y,
                                             gint width,
                                             gint height);
但是改变像素值,会改变图像的样子吗?[/quote]
上面这个函数可以缩放显示吗?
我的图片太大,1024×1024,显示不下了.     

像素值都变了图象当然就变了啊,对图象进行处理不就是对像素值处理吗
回复

使用道具 举报

 楼主| 发表于 2005-8-5 03:34:02 | 显示全部楼层
[quote="yangh"]
你既然是 new_from_file, 如何改的像素值?? 建议试试使用 new_from_pixbuf/bitmap 然后 改了 pixbuf/bitmap, 然后再 set_from_pixbuf/bitmap

我用了pixbuf = gtk_image_get_pixbuf((GtkImage*)image);

其实我是很想用pixbuf/bitmap的,但是我不知道如何用pixbuf/bitmap来创建一个空的512×512,RGB的,8位或16位灰度级的,带alpha同通道(或者不带)的空图片,所以就直接从文件中读了。
要想创建空图片应该怎么用pixbuf/bitmap呢?
回复

使用道具 举报

发表于 2005-8-5 08:55:20 | 显示全部楼层
我自己的一写想法,不知道对不对,楼主看看吧。

如果是要一个空的,可以gdk_pixmap_new (drawarea->window,512,512),创建一个pixmap,然后gdk_pixbuf_get_from_drawable 得到一个pixbuf,这样子 通过gdk_pixbuf_render_threshold_alpha,就可以操作ALPHA通道了。RGB的灰度应该可以通过GC或COLORMAP来操作。通过gdk_draw_pixbuf把pixbuf中的内容写到pixmap,在通过调用gtk_widget_queue_draw_area来调用expose_event事件,在expose_event事件中通过函数gdk_draw_drawable (widget->window,gc, pixmap, .....)完成显示。
如果本身就是要读取图片的,你的办法不是挺好的。
回复

使用道具 举报

发表于 2005-8-5 09:01:29 | 显示全部楼层
[quote:4e749b3db7="xiaocong213"]
上面这个函数可以缩放显示吗?
我的图片太大,1024×1024,显示不下了.     

像素值都变了图象当然就变了啊,对图象进行处理不就是对像素值处理吗[/quote]
不能通过那个函数缩放的,可以用gdk_pixbuf_composite ()
函数,或者自己写代码的。GTK的演示程序中有一个图像缩放的例子的,楼主看看去。
#include <stdlib.h>
#include <gtk/gtk.h>
#include <math.h>

#include "demo-common.h"

#define FRAME_DELAY 50

#define BACKGROUND_NAME "background.jpg"

static const char *image_names[] = {
  "apple-red.png",
  "gnome-applets.png",
  "gnome-calendar.png",
  "gnome-foot.png",
  "gnome-gmush.png",
  "gnome-gimp.png",
  "gnome-gsame.png",
  "gnu-keys.png"
};

#define N_IMAGES G_N_ELEMENTS (image_names)

/* demo window */
static GtkWidget *window = NULL;

/* Current frame */
static GdkPixbuf *frame;

/* Background image */
static GdkPixbuf *background;
static gint back_width, back_height;

/* Images */
static GdkPixbuf *images[N_IMAGES];

/* Widgets */
static GtkWidget *da;

/* Loads the images for the demo and returns whether the operation succeeded */
static gboolean
load_pixbufs (GError **error)
{
  gint i;
  char *filename;

  if (background)
    return TRUE; /* already loaded earlier */

  /* demo_find_file() looks in the the current directory first,
   * so you can run gtk-demo without installing GTK, then looks
   * in the location where the file is installed.
   */
  filename = demo_find_file (BACKGROUND_NAME, error);
  if (!filename)
    return FALSE; /* note that "error" was filled in and returned */

  background = gdk_pixbuf_new_from_file (filename, error);
  g_free (filename);
  
  if (!background)
    return FALSE; /* Note that "error" was filled with a GError */

  back_width = gdk_pixbuf_get_width (background);
  back_height = gdk_pixbuf_get_height (background);

  for (i = 0; i < N_IMAGES; i++)
    {
      filename = demo_find_file (image_names, error);
      if (!filename)
        return FALSE; /* Note that "error" was filled with a GError */
      
      images = gdk_pixbuf_new_from_file (filename, error);
      g_free (filename);
      
      if (!images)
        return FALSE; /* Note that "error" was filled with a GError */
    }

  return TRUE;
}

/* Expose callback for the drawing area */
static gint
expose_cb (GtkWidget          *widget,
           GdkEventExpose *event,
           gpointer           data)
{
  guchar *pixels;
  int rowstride;

  rowstride = gdk_pixbuf_get_rowstride (frame);

  pixels = gdk_pixbuf_get_pixels (frame) + rowstride * event->area.y + event->area.x * 3;

  gdk_draw_rgb_image_dithalign (widget->window,
                                widget->style->black_gc,
                                event->area.x, event->area.y,
                                event->area.width, event->area.height,
                                GDK_RGB_DITHER_NORMAL,
                                pixels, rowstride,
                                event->area.x, event->area.y);

  return TRUE;
}

#define CYCLE_LEN 60

static int frame_num;

/* Timeout handler to regenerate the frame */
static gint
timeout (gpointer data)
{
  double f;
  int i;
  double xmid, ymid;
  double radius;

  gdk_pixbuf_copy_area (background, 0, 0, back_width, back_height,
                        frame, 0, 0);

  f = (double) (frame_num % CYCLE_LEN) / CYCLE_LEN;

  xmid = back_width / 2.0;
  ymid = back_height / 2.0;

  radius = MIN (xmid, ymid) / 2.0;

  for (i = 0; i < N_IMAGES; i++)
    {
      double ang;
      int xpos, ypos;
      int iw, ih;
      double r;
      GdkRectangle r1, r2, dest;
      double k;

      ang = 2.0 * G_PI * (double) i / N_IMAGES - f * 2.0 * G_PI;

      iw = gdk_pixbuf_get_width (images);
      ih = gdk_pixbuf_get_height (images);

      r = radius + (radius / 3.0) * sin (f * 2.0 * G_PI);

      xpos = floor (xmid + r * cos (ang) - iw / 2.0 + 0.5);
      ypos = floor (ymid + r * sin (ang) - ih / 2.0 + 0.5);

      k = (i & 1) ? sin (f * 2.0 * G_PI) : cos (f * 2.0 * G_PI);
      k = 2.0 * k * k;
      k = MAX (0.25, k);

      r1.x = xpos;
      r1.y = ypos;
      r1.width = iw * k;
      r1.height = ih * k;

      r2.x = 0;
      r2.y = 0;
      r2.width = back_width;
      r2.height = back_height;

      if (gdk_rectangle_intersect (&r1, &r2, &dest))
        gdk_pixbuf_composite (images,
                              frame,
                              dest.x, dest.y,
                              dest.width, dest.height,
                              xpos, ypos,
                              k, k,
                              GDK_INTERP_NEAREST,
                              ((i & 1)
                               ? MAX (127, fabs (255 * sin (f * 2.0 * G_PI)))
                               : MAX (127, fabs (255 * cos (f * 2.0 * G_PI)))));
    }

  gtk_widget_queue_draw (da);

  frame_num++;
  return TRUE;
}

static guint timeout_id;

static void
cleanup_callback (GtkObject *object,
                  gpointer   data)
{
  g_source_remove (timeout_id);
  timeout_id = 0;
}

GtkWidget *
do_pixbufs (GtkWidget *do_widget)
{
  if (!window)
    {
      GError *error;

      window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
      gtk_window_set_screen (GTK_WINDOW (window),
                             gtk_widget_get_screen (do_widget));
      gtk_window_set_title (GTK_WINDOW (window), "Pixbufs");
      gtk_window_set_resizable (GTK_WINDOW (window), FALSE);

      g_signal_connect (window, "destroy", G_CALLBACK (gtk_widget_destroyed), &window);
      g_signal_connect (window, "destroy", G_CALLBACK (cleanup_callback), NULL);


      error = NULL;
      if (!load_pixbufs (&error))
        {
          GtkWidget *dialog;

          dialog = gtk_message_dialog_new (GTK_WINDOW (window),
                                           GTK_DIALOG_DESTROY_WITH_PARENT,
                                           GTK_MESSAGE_ERROR,
                                           GTK_BUTTONS_CLOSE,
                                           "Failed to load an image: %s",
                                           error->message);

          g_error_free (error);

          g_signal_connect (dialog, "response",
                            G_CALLBACK (gtk_widget_destroy), NULL);

          gtk_widget_show (dialog);
        }
      else
        {
          gtk_widget_set_size_request (window, back_width, back_height);

          frame = gdk_pixbuf_new (GDK_COLORSPACE_RGB, FALSE, 8, back_width, back_height);

          da = gtk_drawing_area_new ();

          g_signal_connect (da, "expose_event",
                            G_CALLBACK (expose_cb), NULL);

          gtk_container_add (GTK_CONTAINER (window), da);

          timeout_id = g_timeout_add (FRAME_DELAY, timeout, NULL);
        }
    }

  if (!GTK_WIDGET_VISIBLE (window))
    {
      gtk_widget_show_all (window);
    }
  else
    {
      gtk_widget_destroy (window);
      window = NULL;
      g_object_unref (frame);
    }

  return window;
}
回复

使用道具 举报

发表于 2005-8-5 09:02:17 | 显示全部楼层
图片二

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?注册

×
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则

GMT+8, 2024-11-3 00:23 , Processed in 0.069207 second(s), 16 queries .

© 2021 Powered by Discuz! X3.5.

快速回复 返回顶部 返回列表