我正在尝试使用 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;
}
...
}
一些建议:
使用 GDB/Nemvier/Builder 调试崩溃的应用程序,以准确找出问题所在。我使用以下命令行来调试 vala/gtk 应用程序:
G_DEBUG=致命警告 gdb 路径/到/可执行文件
使用内置的 GTK 功能,而不是重新发明它。如果你想使一个小部件成为特定的大小,请使用
Gtk.Widget.set_size_request()
,你不需要上面 90% 的代码。要实现自定义绘图,请创建一个 Gtk.DrawingArea,连接到绘制信号,实现它,你就完成了。