看看GtkWindow如何被生出来的(gtk_window_new()的内部机制).

 

看看GtkWindow如何被生出来的.

Glib/GObject/Gtk也有很长时间了,没有时间往下刨根,现在刨一刨。

 

1 定义GtkWindow.. 2

2 展开它:... 2

gtype.h中定义如下宏:... 2

把定义的宏展开:... 3

再把GtkBin的定义展开:... 4

再展开GtkContainer的定义://不用展开,直接定义的... 5

GtkWidgetget_type()也是直接定义的... 6

GtkObjectget_type()也是直接定义的:... 7

看看GObject的定义:gtk_init()后,这个对象是已经建立了... 8

3 在用的时候,是直接调用一个宏:... 8

下面我们来跟着一下其流程:(步步跟踪其父类,如果父类还没有创建,则先创建父类)    8

Step 1: 程序会调用这个接口:... 8

Step 2: 接着会调用 (层层上访) 9

Step 3:所要先看GtkObject的创立过程:... 9

GtkObject对象存在了,那就看它的子类吧:GtkWidget 10

接着往下搞:GtkWidget对象也有了,在往下看子类:GtkContainer 11

Go on: GtkBin. 12

GtkBin创建后,GtkWindow对象也就创建了;... 13

这样一系列的对象存在如下:... 13

上面仅仅是一些对象,那这些对象携带的回调函数都是从哪里调用的呢?... 14

关键之处是g_object_new()里面调用的函数... 14

重点看看g_object_newv... 14

g_type_class_ref这个就是核心:递归调用... 17

上面的递归调用会调用函数type_class_init_Wm()完成对某个类的初始化,即,分配内存同时调用class_init() 19

Base_class_init()调用的顺序:... 23

1. g_object_base_class_init 23

2 gtk_object_base_class_init 23

3 gtk_widget_base_class_init NULL所以不调用... 23

4 gtk_container_base_class_init 24

5 gtk_bin, gtk_windonwbase_class_init()都是NULL. 24

下面开始class_init: 24

class_init() 之前有个递归过程(图示)... 24

下面用运行代码来检查实际的过程:... 25

条件:... 25

结果:... 25

 

 

 

1 定义GtkWindow

 

G_DEFINE_TYPE_WITH_CODE (GtkWindow, gtk_window, GTK_TYPE_BIN,

                         G_IMPLEMENT_INTERFACE (GTK_TYPE_BUILDABLE,

                                          gtk_window_buildable_interface_init))

 

2 展开它:

gtype.h中定义如下宏:

#define G_DEFINE_TYPE_WITH_CODE(TN, t_n, T_P, _C_)         _G_DEFINE_TYPE_EXTENDED_BEGIN (TN, t_n, T_P, 0) {_C_;} _G_DEFINE_TYPE_EXTENDED_END()

 

 

#define _G_DEFINE_TYPE_EXTENDED_BEGIN(TypeName, type_name, TYPE_PARENT, flags) /

/

static void     type_name##_init              (TypeName        *self); /

static void     type_name##_class_init        (TypeName##Class *klass); /

static gpointer type_name##_parent_class = NULL; /

static void     type_name##_class_intern_init (gpointer klass) /

{ /

  type_name##_parent_class = g_type_class_peek_parent (klass); /

  type_name##_class_init ((TypeName##Class*) klass); /

} /

/

GType /

type_name##_get_type (void) /

{ /

  static volatile gsize g_define_type_id__volatile = 0; /

  if (g_once_init_enter (&g_define_type_id__volatile))  /

    { /

      GType g_define_type_id = /

        g_type_register_static_simple (TYPE_PARENT, /

                                       g_intern_static_string (#TypeName), /

                                       sizeof (TypeName##Class), /

                                       (GClassInitFunc) type_name##_class_intern_init, /

                                       sizeof (TypeName), /

                                       (GInstanceInitFunc) type_name##_init, /

                                       (GTypeFlags) flags); /

      { /* custom code follows */

#define _G_DEFINE_TYPE_EXTENDED_END()  /

        /* following custom code */  /

      }                             /

      g_once_init_leave (&g_define_type_id__volatile, g_define_type_id); /

    }                                 /

  return g_define_type_id__volatile;   /

} /* closes type_name##_get_type() */

 

把定义的宏展开:

static void gtk_window_init (GtkWindow *self);

static void gtk_window_class_init (GtkWindowClass  * klass);

static void gtk_window_parent_class = NULL;

static void gtk_window_class_intern_init (gpointer klass)

{

       gtk_window_parent_class = g_type_class_peek_parent (klass);

       gtk_window_class_init ((GtkWindowClass*) klass);

}

 

GType

gtk_window_get_type (void)

{

       static volatile gsize g_define_type_id__volatile = 0;

       if (g_once_init_enter (&g_define_type_id__volatile))

       {

              GType g_define_type_id =

                     g_type_register_static_simple (GTK_TYPE_BIN, //父亲

                                                        g_intern_static_string ("GtkWindow"),

                                                        sizeof (GtkWindowClass),

                                                        (GClassInitFunc) gtk_window_class_intern_init,

                                                        sizeof (GtkWindow),

                                                        (GInstanceInitFunc) gtk_window_init,

                                                        0);

              {

                     const GInterfaceInfo g_implement_interface = {

                            (GInterfaceInitFunc) gtk_window_buildable_interface_init, NULL, NULL };

 

                     g_type_add_interface_static (g_define_type_id, GTK_TYPE_BUILDABLE, &g_implement_interface);

              }

              g_once_init_leave (&g_define_type_id__volatile, g_define_type_id);

       }

       return g_define_type_id__volatile;

}

 

再把GtkBin的定义展开:

G_DEFINE_ABSTRACT_TYPE (GtkBin, gtk_bin, GTK_TYPE_CONTAINER)

 

static void gtk_bin_init (GtkBin *self);

static void gtk_bin_class_init (GtkBinClass  * klass);

static void gtk_bin_parent_class = NULL;

static void gtk_bin_class_intern_init (gpointer klass)

{

       gtk_bin_parent_class = g_type_class_peek_parent (klass);

       gtk_bin_class_init ((GtkBinClass*) klass);

}

 

GType

gtk_bin_get_type(void)

{

       static volatile gsize g_define_type_id__volatile = 0;

       if (g_once_init_enter (&g_define_type_id__volatile))

       {

              GType g_define_type_id =

                     g_type_register_static_simple (GTK_TYPE_CONTAINER, //GtkBin的父亲

                                                               g_intern_static_string ("GtkBin"),

                                                               sizeof (GtkBinClass),

                                                               (GClassInitFunc) gtk_bin_class_intern_init,

                                                               sizeof (GtkBin),

                                                               (GInstanceInitFunc) gtk_bin_init,

                                                               G_TYPE_FLAG_ABSTRACT);

              {

              }

              g_once_init_leave (&g_define_type_id__volatile, g_define_type_id);

       }

       return g_define_type_id__volatile;

}

 

再展开GtkContainer的定义://不用展开,直接定义的

#define GTK_TYPE_CONTAINER              (gtk_container_get_type ())

/* --- functions --- */

GType

gtk_container_get_type (void)

{

  static GType container_type = 0;

 

  if (!container_type)

    {

      const GTypeInfo container_info =

      {

       sizeof (GtkContainerClass),

       (GBaseInitFunc) gtk_container_base_class_init,

       (GBaseFinalizeFunc) gtk_container_base_class_finalize,

       (GClassInitFunc) gtk_container_class_init,

       NULL        /* class_finalize */,

       NULL        /* class_data */,

       sizeof (GtkContainer),

       0           /* n_preallocs */,

       (GInstanceInitFunc) gtk_container_init,

       NULL,       /* value_table */

      };

 

      static const GInterfaceInfo buildable_info =

      {

       (GInterfaceInitFunc) gtk_container_buildable_init,

       NULL,

       NULL

      };

 

      container_type =

       g_type_register_static (GTK_TYPE_WIDGET, //GtkContainer的父亲

I_("GtkContainer"),

                            &container_info, G_TYPE_FLAG_ABSTRACT);

 

      g_type_add_interface_static (container_type,

                               GTK_TYPE_BUILDABLE,

                               &buildable_info);

 

    }

 

  return container_type;

}

 

GtkWidgetget_type()也是直接定义的

/* --- functions --- */

GType

gtk_widget_get_type (void)

{

  static GType widget_type = 0;

 

  if (G_UNLIKELY (widget_type == 0))

    {

      const GTypeInfo widget_info =

      {

       sizeof (GtkWidgetClass),

       NULL,          /* base_init */

       (GBaseFinalizeFunc) gtk_widget_base_class_finalize,

       (GClassInitFunc) gtk_widget_class_init,

       NULL,          /* class_finalize */

       NULL,          /* class_init */

       sizeof (GtkWidget),

       0,           /* n_preallocs */

       (GInstanceInitFunc) gtk_widget_init,

       NULL,          /* value_table */

      };

 

      const GInterfaceInfo accessibility_info =

      {

       (GInterfaceInitFunc) gtk_widget_accessible_interface_init,

       (GInterfaceFinalizeFunc) NULL,

       NULL /* interface data */

      };

 

      const GInterfaceInfo buildable_info =

      {

       (GInterfaceInitFunc) gtk_widget_buildable_interface_init,

       (GInterfaceFinalizeFunc) NULL,

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
在 Linux 上使用 wxWidgets,并且使用 gtk_plug_new() 函数来嵌入 wxWidgets 应用程序可能会比较复杂,因为 wxWidgets 是一个跨平台的UI框架,它并不直接使用 GTK+ 的特定函数。 如果你希望将 wxWidgets 应用程序嵌入到一个现有的 GTK+ 窗口中,你需要使用 wxGTK 模块提供的函数来创建 wxWidgets 的顶级窗口,并使用 gtk_plug_new() 函数来创建一个插件窗口。然后,你可以通过将插件窗口嵌入到 GTK+ 窗口中,将 wxWidgets 应用程序显示在 GTK+ 窗口中。 以下是一个示例代码,展示了如何在 Linux 上使用 gtk_plug_new() 来嵌入 wxWidgets 应用程序: ```cpp #include <wx/wx.h> #include <gtk/gtk.h> class MyFrame : public wxFrame { public: MyFrame(const wxString& title) : wxFrame(NULL, wxID_ANY, title) { // 创建你的界面元素 // ... } }; class MyApp : public wxApp { public: virtual bool OnInit() override { // 初始化 GTK+ if (!wxApp::OnInit()) return false; gtk_init(nullptr, nullptr); // 创建 GTK+ 窗口 GtkWidget* gtkWindow = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_default_size(GTK_WINDOW(gtkWindow), 800, 600); // 创建插件窗口 GtkWidget* plug = gtk_plug_new(0); gtk_container_add(GTK_CONTAINER(gtkWindow), plug); // 创建 wxWidgets 顶级窗口 MyFrame* frame = new MyFrame("My App"); frame->SetSize(800, 600); frame->Create(plug); // 显示 GTK+ 窗口 gtk_widget_show_all(gtkWindow); return true; } }; wxIMPLEMENT_APP(MyApp); ``` 在这个示例中,我们创建了一个自定义的 wxFrame 类 MyFrame,并在其中添加了一些界面元素。在 MyApp 类的 OnInit() 方法中,我们先调用 wxApp::OnInit() 来初始化 wxWidgets 和 GTK+,然后使用 gtk_window_new() 创建一个 GTK+ 窗口。接下来,我们使用 gtk_plug_new() 创建一个插件窗口,并将它添加到 GTK+ 窗口中。最后,我们创建了一个 MyFrame 实例,并将它的插件窗口设置为刚刚创建的插件窗口。最终,我们显示 GTK+ 窗口。 请注意,这只是一个简单的示例,实际上在将 wxWidgets 应用程序嵌入到 GTK+ 窗口中时还可能涉及到其他复杂的问题。如果你需要更多的控制权和灵活性,可能需要深入研究 wxWidgets 和 GTK+ 的文档,并根据自己的需求进行适当的调整和定制。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值