0

私は自分のRGB画像を取得し、GTK+-2.2を使用してそれらをウィンドウに表示できるクラスを作成しています。私のImageクラスは、画像をパックされた24ビットRGBバイトとして格納するため、変換は簡単です。gdk_draw_rgb(...)メソッドを使用してウィンドウに描画していますが、何も描画されません。ウィンドウは灰色で表示されます。

私はこれをCairoを使用して機能させましたが、残念ながらCairoは32bpp形式の画像しか表現できず、その変換を行うのは遅すぎました。

  class ImageDisplay
  {
    public:
      ImageDisplay();
      ~ImageDisplay();

      void showImage(Image img, std::string label="");

    private:
      std::thread _gtkThread;

      std::map<std::string, GtkWidget*> _windows;
  };

// ######################################################################
void gtkThreadMethod()
{
  g_thread_init(NULL);
  gdk_threads_init();
  gdk_threads_enter();

  int argc=1;
  char **argv = new char*;
  argv[0] = new char[8];
  sprintf(argv[0], "display");
  gtk_init(&argc, &argv);

  gdk_rgb_set_verbose(TRUE);

  gtk_main();

  gdk_threads_leave();
}

// ######################################################################
ImageDisplay::ImageDisplay()
{ 
  // Start gtk in its own thread
  _gtkThread = std::thread(gtkThreadMethod);
}

// ######################################################################
ImageDisplay::~ImageDisplay()
{  
  // Tell GTK that it's time to quit
  gdk_threads_enter();
  gtk_main_quit();
  gdk_threads_leave();

  // Wait for the thread to die
  _gtkThread.join();
}

// ######################################################################
void ImageDisplay::showImage(Image img, std::string label)
{
  gdk_threads_enter();

  // Create a new window if one doesn't yet exist
  if(_windows.find(label) == _windows.end())
  {
    GtkWidget* window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_window_set_title(GTK_WINDOW(window), label.c_str());
    gtk_window_set_default_size(GTK_WINDOW(window), img.dims().w(), img.dims().h());
    gtk_widget_set_app_paintable(window, TRUE);
    gtk_window_set_resizable(GTK_WINDOW(window), true);
    GdkGeometry size_hints;
    size_hints.min_aspect = 1;
    size_hints.max_aspect = 1;
    gtk_window_set_geometry_hints(GTK_WINDOW(window), window,
        &size_hints, GDK_HINT_ASPECT);
    gtk_widget_show_all(window);
    _windows[label] = window;
  }

  GtkWidget* window = _windows[label];

  GdkGC *gc = gdk_gc_new(gtk_widget_get_root_window(window));
  gdk_draw_rgb_image(
      gtk_widget_get_root_window(window),
      gc,
      0, 0,
      img.dims().w(), img.dims().h(),
      GDK_RGB_DITHER_NORMAL, 
      (const unsigned char*)img.const_begin(), 
      img.dims().w()*3);

  gdk_threads_leave();
}
4

2 に答える 2

1

GtkImage(クライアント側の) 画像を表示するには、カスタム ペインティングを行うためにランダムなウィジェットを「オーバーロード」するのではなく、ウィジェットの使用を検討する必要があると思います。

GdkPixbufこれにより、ピクセルを保持しているが公開されます。

于 2010-11-09T09:41:45.947 に答える
1
gdk_draw_rgb_image(
  gtk_widget_get_root_window(window),

問題があります。X 用語 (GTK+ が多用している) では、「ルート ウィンドウ」はデスクトップの背景を指します。あなたはあなたのウィンドウを表すgtk_widget_get_windowものを手に入れたいと思っています。GdkDrawable

しかし...私は上記の行からスタックをそれほど遠くまで歩いていません。このコードの呼び出し元がどのように見えるかはわかりませんが、通常はすぐにではなく「公開イベント」ハンドラーで描画したいと考えています電話した後gtk_window_new。私がこの種のコードを最後に書いたとき (かなり時間が経っていますが、認めます)、GdkPixmap描画する を作成し、その内容をGdkWindowエクスポーズ イベントでユーザーに表示されるものにコピーしていました。ここGtkDrawingAreaではウィジェットが役立つので、それを使用して例を検索します。

于 2010-11-09T09:21:12.327 に答える