Нарисуйте фоновое изображение в gtk - ни одна из моих попыток не сработала - PullRequest
4 голосов
/ 29 сентября 2011

Я безуспешно пытался установить фоновое изображение в виджете GTK, даже после попытки 4 различных подходов.

Следующая программа содержит 3 подхода (4-й подход не требует кода). Я скомпилировал его, используя MinGW (g ++ 4.5.0) и gtkmm 2.4. Макрос APPROACH может быть установлен в 1, 2 или 3, чтобы выбрать подход к компиляции. Я также добавил ссылки в комментарии, чтобы вы могли узнать, откуда у меня идеи.

#include <iostream>
#include <gtkmm/main.h>
#include <gtkmm/alignment.h>
#include <gtkmm/box.h>
#include <gtkmm/entry.h>
#include <gtkmm/eventbox.h>
#include <gtkmm/frame.h>
#include <gtkmm/image.h>
#include <gtkmm/label.h>
#include <gtkmm/table.h>
#include <gtkmm/window.h>

// Set this to 1, 2 or 3 to try different ways of drawing the background
// Set to 0 to load no background at all
#define APPROACH (0)

// Making this alignment global in order to modify it from drawBackground
Gtk::Alignment* alignment;

bool drawBackground(GdkEventExpose* event) {
    std::cout << "Draw background" << std::endl;

    // Load background image
    Glib::RefPtr<Gdk::Pixbuf> pixbuf = Gdk::Pixbuf::create_from_file("background.jpg");
    Glib::RefPtr<Gdk::Pixmap> pixmap;
    Glib::RefPtr<Gdk::Bitmap> mask;
    pixbuf->render_pixmap_and_mask(pixmap, mask,0);

    {
        // Test that pixbuf was created correctly
        Glib::RefPtr<Gdk::Pixbuf> back_to_pixbuf = Gdk::Pixbuf::create((Glib::RefPtr<Gdk::Drawable>)pixmap, 0, 0, pixbuf->get_width(), pixbuf->get_height());
        back_to_pixbuf->save("back_to_pixbuf.png", "png");
    }

#if APPROACH == 1
    // Approach 1: draw_pixbuf
    // Ref: http://islascruz.org/html/index.php/blog/show/Image-as-background-in-a-Gtk-Application..html
    Glib::RefPtr<Gtk::Style> style = alignment->get_style();
    alignment->get_window()->draw_pixbuf(style->get_bg_gc(Gtk::STATE_NORMAL), pixbuf, 0, 0, 0, 200, pixbuf->get_width(), pixbuf->get_height(), Gdk::RGB_DITHER_NONE, 0, 0);
#endif

#if APPROACH == 2
    // Approach 2: set_back_pixmap
    // Ref: http://www.gtkforums.com/viewtopic.php?t=446
    //      /3195258/gtk-drawing-set-fonovoe-izobrazhenie
    alignment->get_window()->set_back_pixmap(pixmap);
#endif
}

int main (int argc, char *argv[])
{
    Gtk::Main kit(argc, argv);

    Gtk::Window w;
    Gtk::VBox mainBox;

    // Top image
    Gtk::Image topImage("header.jpg");
    mainBox.pack_start(topImage,false,false,0);

    // Middle alignment
    alignment = Gtk::manage(new Gtk::Alignment);
    mainBox.pack_start(*alignment,true,true,0);

    // Create widget
    Gtk::Alignment mywidget(0.5, 0.5, 0.1, 0.9);
    Gtk::Table table;
    Gtk::Label label1("Username"); table.attach(label1,0,1,0,1);
    Gtk::Label label2("Password"); table.attach(label2,0,1,1,2);
    Gtk::Entry entry1;             table.attach(entry1,1,2,0,1);
    Gtk::Entry entry2;             table.attach(entry2,1,2,1,2);
    Gtk::Button button("Login");   table.attach(button,1,2,2,3);
    mywidget.add(table);

    // Put widget in middle alignment
    alignment->add(mywidget);

    // Try to change the background
#if APPROACH == 1 || APPROACH == 2
    alignment->signal_expose_event().connect(sigc::ptr_fun(&drawBackground), true);
#endif

#if APPROACH == 3
    // Approach 3: modify the style using code
    // Ref: http://www.gtkforums.com/viewtopic.php?t=446
    // Load background image
    Glib::RefPtr<Gdk::Pixbuf> pixbuf = Gdk::Pixbuf::create_from_file("background.jpg");
    Glib::RefPtr<Gdk::Pixmap> pixmap;
    Glib::RefPtr<Gdk::Bitmap> mask;
    pixbuf->render_pixmap_and_mask(pixmap, mask,0);
    Glib::RefPtr<Gtk::Style> style = alignment->get_style()->copy();
    style->set_bg_pixmap(Gtk::STATE_NORMAL,pixmap);
    style->set_bg_pixmap(Gtk::STATE_ACTIVE,pixmap);
    style->set_bg_pixmap(Gtk::STATE_PRELIGHT,pixmap);
    style->set_bg_pixmap(Gtk::STATE_SELECTED,pixmap);
    style->set_bg_pixmap(Gtk::STATE_INSENSITIVE,pixmap);
    alignment->set_style(style);
#endif

    // Approach 4: modify share\themes\MS-Windows\gtk-2.0
    // adding the following line
    // bg_pixmap[NORMAL] = "D:\\path\\to\\file\\background.jpg"
    // in the style "msw-default" section
    // Ref: http://lists.ximian.com/pipermail/gtk-sharp-list/2005-August/006324.html

    // Show the window
    w.add(mainBox);
    w.show_all();
    kit.run(w);
    return 0;
}

Ссылки на изображения, которые я использовал: header.jpg background.jpg

Макет имитирует мою настоящую программу. Главное окно содержит Gtk :: VBox с изображением заголовка вверху и Gtk :: Alignment внизу. Содержимое этого выравнивания со временем будет меняться, но я хочу, чтобы фоновое изображение всегда было видно.

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

enter image description here

Подход 1 - это тот, который ближе к работе, хотя он скрывает метки и кнопки:

enter image description here

Подходы 2 и 3 выглядят так же, как загрузка без фона. Кроме того, подход 2 дает мне следующее сообщение об ошибке:

(test-img-fondo.exe:1752): Gdk-CRITICAL **: gdk_window_set_back_pixmap: assertion `pixmap == NULL || !parent_relative' failed

Наконец, в подходе 4 я пытаюсь изменить общий ресурс \ themes \ MS-Windows \ gtk-2.0, добавив следующую строку

bg_pixmap[NORMAL] = "D:\\path\\to\\file\\background.jpg"

в разделе «msw-default». Это тоже не работает.

Итак, кто-нибудь успешно нарисовал фоновое изображение на виджете Gtk? Это вообще возможно? Какие-нибудь изменения в моем коде, которые сделали бы эту работу? Есть обходные пути?

Вся помощь очень ценится.

1 Ответ

7 голосов
/ 29 сентября 2011

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

alignment->signal_expose_event().connect(sigc::ptr_fun(&drawBackground), true);

для этого:

alignment->signal_expose_event().connect(sigc::ptr_fun(&drawBackground), false);

Таким образом, вызов drawBackground происходит до того, как gtk вызывает свои собственные обработчики.

enter image description here

Следует также отметить, что в реальной программе изображения должны загружаться один раз за пределы drawBackground.

...