带有 Vala 的自定义 GTK 小部件



我正在尝试使用 Vala 创建一个简单的自定义 GTK 小部件:

public class PageView : Gtk.Widget {
    public PageView () {
        //base ();
        set_name ("pageview");
        set_has_window (true);
    }

    /*
     *              Method and Signal Overrides
     */
    public override Gtk.SizeRequestMode get_request_mode () {
        return Gtk.SizeRequestMode.CONSTANT_SIZE;
            // Don’t trade height-for-width or width-for-height
    }
    public override void get_preferred_width
                                               (out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_widthn");
        minimum = natural = pg_pixel_width;
    }
    public override void get_preferred_height
                                               (out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_heightn");
        minimum = natural = pg_pixel_height;
    }
    public override void get_preferred_width_for_height
                                               (    int height,
                                                out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_width_for_heightn");
        minimum = natural = pg_pixel_width;
    }
    public override void get_preferred_height_for_width
                                               (    int width,
                                                out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_height_for_widthn");
        minimum = natural = pg_pixel_height;
    }

    public override void size_allocate (Gtk.Allocation alloc) {
        stdout.printf ("PageView#size_allocaten");
        set_allocation (alloc);
        if (get_window() != null) {
            get_window().move_resize (
                alloc.x, alloc.y, alloc.width, alloc.height);
        }
    }
    public override void realize () {
        stdout.printf ("PageView#realizen");
        set_realized (true);
        if (get_window() == null) {
            Gtk.Allocation allocation;
            get_allocation (out allocation);
            _window_attr = Gdk.WindowAttr () {
                x = allocation.x,
                y = allocation.y,
                width = allocation.width,
                height = allocation.height,
                event_mask = get_events() | Gdk.EventMask.EXPOSURE_MASK,
                window_type = Gdk.WindowType.CHILD,
                wclass = Gdk.WindowWindowClass.INPUT_OUTPUT
            };
            _window_attr_type =  Gdk.WindowAttributesType.X |
                                 Gdk.WindowAttributesType.Y;
            _window = new Gdk.Window (
                get_parent_window (), _window_attr, _window_attr_type);
            set_window (_window);
        }
    }
    public override void unrealize () {
        stdout.printf ("PageView#unrealizen");
    }

    public override bool draw (Cairo.Context cr) {
        stdout.printf ("PageView#drawn");
        Gtk.Allocation allocation;
        get_allocation (out allocation);
        get_style_context () .render_background (cr,
            allocation.x,     allocation.y,
            allocation.width, allocation.height);
        cr.save ();
            cr.scale (allocation.width, allocation.height);
            cr.set_source_rgba (255, 255, 0, 1.0);
            cr.set_line_width (10);
            cr.line_to (1, 1);
            cr.stroke ();
        cr.restore ();

        return true;
    }

    /*
     *                  Public Attributes
     */
    public int pg_pixel_width {
        get { return 480; }
    }
    public int pg_pixel_height {
        get { return 480; }
    }

    /*
     *                  Private Members
     */
    private Gdk.Window      _window;
    private Gdk.WindowAttr  _window_attr;
    private Gdk.WindowAttributesType _window_attr_type;
}

问题是,当我将其添加到我的主 Gtk.Window 时,我遇到了分段错误。这些是我收到的调试消息:

PageView#get_preferred_height
PageView#get_preferred_width
PageView#size_allocate
PageView#realize
PageView#get_preferred_height
PageView#get_preferred_width
PageView#size_allocate
PageView#size_allocate
Segmentation fault (core dumped)

似乎如果我将 realize () 内部对 set_window (_window) 的调用更改为 set_window(null) ,或者如果我将空父窗口传递给新创建的 Gdk.Window,应用程序将运行而没有分段错误(但小部件在这两种情况下都不会按预期显示)。我主要按照此示例来实现虚拟方法,并尝试在 Vala 中移植 C 代码。问题的原因可能是什么?

最后,我从构造函数中删除了set_has_window(true),只实现了draw()信号。这似乎已经成功了!下面是工作代码段:

public class PageView : Gtk.Widget {
    public PageView () {
        set_name ("pageview");
    }

    /*
     *              Method and Signal Overrides
     */
    public override Gtk.SizeRequestMode get_request_mode () {
        return Gtk.SizeRequestMode.CONSTANT_SIZE;
            // Don’t trade height-for-width or width-for-height
    }
    public override void get_preferred_width
                                               (out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_widthn");
        minimum = natural = pg_pixel_width;
    }
    public override void get_preferred_height
                                               (out int minimum,
                                                out int natural) {
        stdout.printf ("PageView#get_preferred_heightn");
        minimum = natural = pg_pixel_height;
    }
    public override void size_allocate (Gtk.Allocation alloc) {
        stdout.printf ("PageView#size_allocaten");
        base.size_allocate (alloc);
    }
    public override void realize () {
        stdout.printf ("PageView#realizen");
        base.realize ();
    }
    public override void unrealize () {
        stdout.printf ("PageView#unrealizen");
        base.unrealize ();
    }

    public override bool draw (Cairo.Context cr) {
        stdout.printf ("PageView#drawn");
        Gtk.Allocation allocation;
        get_allocation (out allocation);
        cr.set_line_width (1);
        cr.set_source_rgba (255, 255, 0, 1);    
        cr.save ();
            cr.scale (allocation.width, allocation.height);
            cr.move_to (0, 0);
            cr.line_to (1, 1);
        cr.restore ();
        cr.stroke ();
        return false;
    }
    ...
}
这里有

一些建议:

  1. 使用 GDB/Nemvier/Builder 调试崩溃的应用程序,以准确找出问题所在。我使用以下命令行来调试 vala/gtk 应用程序:

    G_DEBUG=致命警告 gdb 路径/到/可执行文件

  2. 使用内置的 GTK 功能,而不是重新发明它。如果你想使一个小部件成为特定的大小,请使用 Gtk.Widget.set_size_request() ,你不需要上面 90% 的代码。要实现自定义绘图,请创建一个 Gtk.DrawingArea,连接到绘制信号,实现它,你就完成了。

相关内容

  • 没有找到相关文章

最新更新