2022-12-21 14:31:33 +01:00
Up: [README.md ](../README.md ), Prev: [Section 10 ](sec10.md ), Next: [Section 12 ](sec12.md )
2021-01-11 03:15:04 +01:00
2022-12-21 14:31:33 +01:00
# Instance Initialization and destruction
2020-12-21 13:12:05 +01:00
2021-06-17 13:04:16 +02:00
A new version of the text file editor (`tfe`) will be made in this section and the following four sections.
It is `tfe5` .
There are many changes from the prior version.
2021-06-30 16:52:33 +02:00
They are located in two directories, [src/tfe5 ](../src/tfe5 ) and [src/tfetextview ](../src/tfetextview ).
2020-12-21 13:12:05 +01:00
2021-06-15 04:24:42 +02:00
## Encapsulation
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
We've divided C source file into two parts.
But it is not enough in terms of encapsulation.
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
- `tfe.c` includes everything other than TfeTextView.
It should be divided at least into two parts, `tfeapplication.c` and `tfenotebook.c` .
- Header files also need to be organized.
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
However, first of all, I'd like to focus on the object TfeTextView.
It is a child object of GtkTextView and has a new member `file` in it.
The important thing is to manage the Gfile object pointed by `file` .
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
- What is necessary to GFile when creating (or initializing) TfeTextView?
- What is necessary to GFile when destructing TfeTextView?
- TfeTextView should read/write a file by itself or not?
- How it communicates with objects outside?
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
You need to know at least class, instance and signals before thinking about them.
I will explain them in this section and the next section.
After that I will explain:
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
- Organizing functions.
- How to use GtkFileChooserDialog
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
## GObject and its children
2020-12-22 03:30:06 +01:00
2022-12-21 14:31:33 +01:00
GObject and its children are objects, which have both class and object C structures.
First, think about instances.
An instance is memories which has the object structure.
The following is the structure of TfeTextView.
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
~~~C
/* This typedef statement is automatically generated by the macro G_DECLARE_FINAL_TYPE */
typedef struct _TfeTextView TfeTextView;
struct _TfeTextView {
GtkTextView parent;
GFile *file;
};
~~~
The members of the structure are:
2021-06-17 13:04:16 +02:00
- The type of `parent` is GtkTextView which is C structure.
It is declared in `gtktextview.h` .
GtkTextView is the parent of TfeTextView.
2022-12-21 14:31:33 +01:00
- `file` is a pointer to a GFile. It can be NULL if no file corresponds to the TfeTextView instance.
2021-06-15 04:24:42 +02:00
2022-12-21 14:31:33 +01:00
You can find the declaration of the ancestors' object structures in the source files of GTK and GLib.
The following is extracted from the source files (not exactly the same).
2020-12-22 03:30:06 +01:00
2021-01-25 13:20:15 +01:00
~~~C
2021-06-15 04:24:42 +02:00
typedef struct _GObject GObject;
typedef struct _GObject GInitiallyUnowned;
struct _GObject
{
GTypeInstance g_type_instance;
volatile guint ref_count;
GData *qdata;
};
typedef struct _GtkWidget GtkWidget;
struct _GtkWidget
{
GInitiallyUnowned parent_instance;
GtkWidgetPrivate *priv;
2021-01-25 13:20:15 +01:00
};
2020-12-22 03:30:06 +01:00
2021-06-15 04:24:42 +02:00
typedef struct _GtkTextView GtkTextView;
struct _GtkTextView
{
GtkWidget parent_instance;
GtkTextViewPrivate *priv;
};
2021-01-25 13:20:15 +01:00
~~~
2020-12-22 03:30:06 +01:00
2021-06-17 13:04:16 +02:00
In each structure, its parent is declared at the top of the members.
2022-12-21 14:31:33 +01:00
So, all the ancestors are included in the child object.
2021-06-15 04:24:42 +02:00
The structure of `TfeTextView` is like the following diagram.
![The structure of the instance TfeTextView ](../image/TfeTextView.png )
2022-12-21 14:31:33 +01:00
Derivable classes (ancestor classes) have their own private data area which are not included by the structure above.
For example, GtkWidget has GtkWidgetPrivate (C structure) for its private data.
Notice declarations are not definitions.
So, no memories are allocated when C structures are declared.
Memories are allocated to them from the heap area when the `tfe_text_view_new` function is called.
At the same time, the ancestors' private area allocated for the TfeTetView.
They are hidden from TfeTextView and it can't access to them directly.
The created memory is called instance.
When a TfeTextView instance is created, it is given three data area.
- The instance (C structure).
- GtkWidgetPrivate structure.
- GtkTextViewPrivate structure.
2021-06-15 04:24:42 +02:00
2022-12-21 14:31:33 +01:00
TfeTextView functions can access to its instance only.
The GtkWidgetPrivate and GtkTextViewPrivate are used by the ancestors' functions.
See the following example.
~~~C
GtkWidget *tv = tfe_text_view_new ();
GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
~~~
The parent's function `gtk_text_view_get_buffer` accesses the GtkTextViewPrivate data (owned by `tv` ).
There is a pointer, which points the GtkBuffer, in the private area and the function returns the pointer.
(Actual behavior is a bit more complicated.)
TfeTextView instances inherit the ancestors functions like this.
A TfeTextView instance is created every time the `tfe_text_view_new` function is called.
Therefore, multiple TfeTextView instances can exist.
## Initialization of TfeTextView instances
2021-06-15 04:24:42 +02:00
2021-06-17 13:04:16 +02:00
The function `tfe_text_view_new` creates a new TfeTextView instance.
2020-12-21 13:12:05 +01:00
2021-01-25 10:35:49 +01:00
~~~C
2021-06-15 04:24:42 +02:00
1 GtkWidget *
2 tfe_text_view_new (void) {
2022-12-21 14:31:33 +01:00
3 return GTK_WIDGET (g_object_new (TFE_TYPE_TEXT_VIEW, "wrap-mode", GTK_WRAP_WORD_CHAR, NULL));
2021-06-15 04:24:42 +02:00
4 }
2021-01-25 10:35:49 +01:00
~~~
2021-01-13 05:29:35 +01:00
2022-09-14 14:02:11 +02:00
When this function is invoked, a TfeTextView instance is created and initialized.
2021-06-18 15:35:20 +02:00
The initialization process is as follows.
2021-06-15 04:24:42 +02:00
2022-12-21 14:31:33 +01:00
1. When the instance is created, GtkWidgetPrivate and GtkTextViewPrivate structures are also created
2. Initializes GObject (GInitiallyUnowned) part in the TfeTextView instance.
3. Initializes GtkWidget part (the first `priv` ) in the TfeTextView instance and GtkWidgetPrivate structure.
4. Initializes GtkTextView part (the second `priv` ) in the TfeTextView instance and GtkTextViewPrivate structure.
5. Initializes TfeTextView part (`file`) in the TfeTextView instance.
2021-06-15 04:24:42 +02:00
2022-12-21 14:31:33 +01:00
The step two through four is done by `g_object_init` , `gtk_widget_init` and `gtk_text_view_init` .
2021-06-18 15:35:20 +02:00
They are called by the system automatically and you don't need to care about them.
Step four is done by the function `tfe_text_view_init` in `tfetextview.c` .
2021-01-13 05:29:35 +01:00
2021-01-25 13:20:15 +01:00
~~~C
2021-06-15 04:24:42 +02:00
1 static void
2 tfe_text_view_init (TfeTextView *tv) {
3 tv->file = NULL;
4 }
2021-01-25 13:20:15 +01:00
~~~
2021-01-13 05:29:35 +01:00
2021-06-15 04:24:42 +02:00
This function just initializes `tv->file` to be `NULL` .
## Functions and Classes
In Gtk, all objects derived from GObject have class and instance (except abstract object).
2022-12-21 14:31:33 +01:00
Instances are memory of C structure, which are described in the previous two subsections.
2021-06-15 04:24:42 +02:00
Each object can have more than one instance.
Those instances have the same structure.
2022-12-21 14:31:33 +01:00
Instances just have data.
Therefore, it doesn't define object's behavior.
2021-06-15 04:24:42 +02:00
We need at least two things.
2022-12-21 14:31:33 +01:00
One is functions and the other is class methods.
2021-06-15 04:24:42 +02:00
You've already seen many functions.
2022-12-21 14:31:33 +01:00
For example,
- `TfeTextView *tfe_text_view_new (void);` is a function to create a TfeTextView instance.
- `GtkTextBuffer *gtk_text_view_get_buffer (GtkTextView *textview)` is a function to get a GtkTextBuffer from GtkTextView.
2021-06-15 04:24:42 +02:00
Functions are public, which means that they are expected to be used by other objects.
2022-12-21 14:31:33 +01:00
They are similar to public methods in object oriented languages.
2021-01-13 05:29:35 +01:00
2022-12-21 14:31:33 +01:00
Class (C structure) mainly consists of pointers to functions.
The functions are called class methods and used by the object itself or its descendant objects.
2021-06-15 04:24:42 +02:00
For example, GObject class is declared in `gobject.h` in GLib source files.
2021-01-13 05:29:35 +01:00
2021-01-25 13:20:15 +01:00
~~~C
2021-06-15 04:24:42 +02:00
1 typedef struct _GObjectClass GObjectClass;
2 typedef struct _GObjectClass GInitiallyUnownedClass;
3
2021-06-17 13:04:16 +02:00
4 struct _GObjectClass
5 {
6 GTypeClass g_type_class;
2022-12-21 14:31:33 +01:00
7
8 /*< private > */
9 GSList *construct_properties;
10
11 /*< public > */
12 /* seldom overridden */
13 GObject* (*constructor) (GType type,
14 guint n_construct_properties,
15 GObjectConstructParam *construct_properties);
16 /* overridable methods */
17 void (*set_property) (GObject *object,
18 guint property_id,
19 const GValue *value,
20 GParamSpec *pspec);
21 void (*get_property) (GObject *object,
22 guint property_id,
23 GValue *value,
24 GParamSpec *pspec);
25 void (*dispose) (GObject *object);
26 void (*finalize) (GObject *object);
27 /* seldom overridden */
28 void (*dispatch_properties_changed) (GObject *object,
29 guint n_pspecs,
30 GParamSpec **pspecs);
31 /* signals */
32 void (*notify) (GObject *object,
33 GParamSpec *pspec);
34
35 /* called when done constructing */
36 void (*constructed) (GObject *object);
37
38 /*< private > */
39 gsize flags;
40
41 gsize n_construct_properties;
42
43 gpointer pspecs;
44 gsize n_pspecs;
45
46 /* padding */
47 gpointer pdummy[3];
48 };
49
2021-01-25 13:20:15 +01:00
~~~
2021-01-13 05:29:35 +01:00
2021-06-17 13:04:16 +02:00
There's a pointer to the function `dispose` in line 23.
2021-01-13 05:29:35 +01:00
2021-06-15 04:24:42 +02:00
~~~C
void (*dispose) (GObject *object);
~~~
2021-02-06 09:26:57 +01:00
2021-06-15 04:24:42 +02:00
The declaration is a bit complicated.
The asterisk before the identifier `dispose` means pointer.
So, the pointer `dispose` points to a function which has one parameter, which points a GObject structure, and returns no value.
2021-06-17 13:04:16 +02:00
In the same way, line 24 says `finalize` is a pointer to the function which has one parameter, which points a GObject structure, and returns no value.
2021-01-13 05:29:35 +01:00
2021-01-25 13:20:15 +01:00
~~~C
2021-06-15 04:24:42 +02:00
void (*finalize) (GObject *object);
~~~
Look at the declaration of `_GObjectClass` so that you would find that most of the members are pointers to functions.
2022-12-21 14:31:33 +01:00
- 13: A function pointed by `constructor` is called when the instance is generated. It completes the initialization of the instance.
- 25: A function pointed by `dispose` is called when the instance destructs itself.
2021-06-15 04:24:42 +02:00
Destruction process is divided into two phases.
The first one is called disposing.
In this phase, the instance releases all the references to other instances.
The second phase is finalizing.
2022-12-21 14:31:33 +01:00
- 26: A function pointed by `finalize` finishes the destruction process.
2021-06-15 04:24:42 +02:00
- The other pointers point to functions which are called while the instance lives.
2021-06-17 13:04:16 +02:00
These functions are called class methods.
The methods are open to its descendants.
But not open to the objects which are not the descendants.
2021-06-15 04:24:42 +02:00
## TfeTextView class
2022-12-21 14:31:33 +01:00
TfeTextView class is a structure and it includes all its ancestors' classes in it.
Therefore, classes have similar hierarchy to instances.
2021-06-18 15:35:20 +02:00
2022-12-21 14:31:33 +01:00
~~~
GObjectClass (GInitiallyUnownedClass) -- GtkWidgetClass -- GtkTextViewClass -- TfeTextViewClass
2021-06-18 15:35:20 +02:00
~~~
2022-12-21 14:31:33 +01:00
The following is extracted from the source codes (not exactly the same).
2021-06-15 04:24:42 +02:00
~~~C
2021-06-17 13:04:16 +02:00
1 struct _GtkWidgetClass
2 {
3 GInitiallyUnownedClass parent_class;
4
5 /*< public > */
6
7 /* basics */
8 void (* show) (GtkWidget *widget);
9 void (* hide) (GtkWidget *widget);
10 void (* map) (GtkWidget *widget);
11 void (* unmap) (GtkWidget *widget);
12 void (* realize) (GtkWidget *widget);
13 void (* unrealize) (GtkWidget *widget);
14 void (* root) (GtkWidget *widget);
15 void (* unroot) (GtkWidget *widget);
16 void (* size_allocate) (GtkWidget *widget,
17 int width,
18 int height,
19 int baseline);
20 void (* state_flags_changed) (GtkWidget *widget,
21 GtkStateFlags previous_state_flags);
22 void (* direction_changed) (GtkWidget *widget,
23 GtkTextDirection previous_direction);
24
25 /* size requests */
26 GtkSizeRequestMode (* get_request_mode) (GtkWidget *widget);
27 void (* measure) (GtkWidget *widget,
28 GtkOrientation orientation,
29 int for_size,
30 int *minimum,
31 int *natural,
32 int *minimum_baseline,
33 int *natural_baseline);
34
35 /* Mnemonics */
36 gboolean (* mnemonic_activate) (GtkWidget *widget,
37 gboolean group_cycling);
38
39 /* explicit focus */
40 gboolean (* grab_focus) (GtkWidget *widget);
41 gboolean (* focus) (GtkWidget *widget,
42 GtkDirectionType direction);
43 void (* set_focus_child) (GtkWidget *widget,
44 GtkWidget *child);
45
46 /* keyboard navigation */
47 void (* move_focus) (GtkWidget *widget,
48 GtkDirectionType direction);
49 gboolean (* keynav_failed) (GtkWidget *widget,
50 GtkDirectionType direction);
51
52 gboolean (* query_tooltip) (GtkWidget *widget,
53 int x,
54 int y,
55 gboolean keyboard_tooltip,
56 GtkTooltip *tooltip);
57
58 void (* compute_expand) (GtkWidget *widget,
59 gboolean *hexpand_p,
60 gboolean *vexpand_p);
61
62 void (* css_changed) (GtkWidget *widget,
63 GtkCssStyleChange *change);
64
65 void (* system_setting_changed) (GtkWidget *widget,
66 GtkSystemSetting settings);
67
68 void (* snapshot) (GtkWidget *widget,
69 GtkSnapshot *snapshot);
70
71 gboolean (* contains) (GtkWidget *widget,
72 double x,
73 double y);
74
75 /*< private > */
76
77 GtkWidgetClassPrivate *priv;
78
79 gpointer padding[8];
80 };
81
82 struct _GtkTextViewClass
83 {
84 GtkWidgetClass parent_class;
85
86 /*< public > */
87
88 void (* move_cursor) (GtkTextView *text_view,
89 GtkMovementStep step,
90 int count,
91 gboolean extend_selection);
92 void (* set_anchor) (GtkTextView *text_view);
93 void (* insert_at_cursor) (GtkTextView *text_view,
94 const char *str);
95 void (* delete_from_cursor) (GtkTextView *text_view,
96 GtkDeleteType type,
97 int count);
98 void (* backspace) (GtkTextView *text_view);
99 void (* cut_clipboard) (GtkTextView *text_view);
100 void (* copy_clipboard) (GtkTextView *text_view);
101 void (* paste_clipboard) (GtkTextView *text_view);
102 void (* toggle_overwrite) (GtkTextView *text_view);
103 GtkTextBuffer * (* create_buffer) (GtkTextView *text_view);
104 void (* snapshot_layer) (GtkTextView *text_view,
2022-12-21 14:31:33 +01:00
105 GtkTextViewLayer layer,
106 GtkSnapshot *snapshot);
2021-06-17 13:04:16 +02:00
107 gboolean (* extend_selection) (GtkTextView *text_view,
108 GtkTextExtendSelection granularity,
109 const GtkTextIter *location,
110 GtkTextIter *start,
111 GtkTextIter *end);
112 void (* insert_emoji) (GtkTextView *text_view);
113
114 /*< private > */
115
116 gpointer padding[8];
117 };
118
119 /* The following definition is generated by the macro G_DECLARE_FINAL_TYPE */
120 typedef struct {
121 GtkTextView parent_class;
122 } TfeTextViewClass;
123
2021-06-15 04:24:42 +02:00
~~~
2021-06-17 13:04:16 +02:00
- 120-122: This three lines are generated by the macro `G_DECLARE_FINAL_TYPE` .
2021-06-15 04:24:42 +02:00
So, they are not written in either `tfe_text_view.h` or `tfe_text_view.c` .
2022-12-21 14:31:33 +01:00
- 3, 84, 121: Each class has its parent class at the first member of its structure.
2021-06-15 04:24:42 +02:00
It is the same as instance structures.
- Class members in ancestors are open to the descendant class.
So, they can be changed in `tfe_text_view_class_init` function.
2022-12-21 14:31:33 +01:00
For example, the `finalize` pointer in GObjectClass will be overridden later in `tfe_text_view_class_init` .
2021-06-15 04:24:42 +02:00
(Override is an object oriented programming terminology.
Override is rewriting ancestors' class methods in the descendant class.)
- Some class methods are often overridden.
`set_property` , `get_property` , `dispose` , `finalize` and `constructed` are such methods.
TfeTextViewClass includes its ancestors' class in it.
It is illustrated in the following diagram.
![The structure of TfeTextView Class ](../image/TfeTextViewClass.png )
## Destruction of TfeTextView
Every Object derived from GObject has a reference count.
If an object A refers to an object B, then A keeps a pointer to B in A and at the same time increases the reference count of B by one with the function `g_object_ref (B)` .
If A doesn't need B any longer, then A discards the pointer to B (usually it is done by assigning NULL to the pointer) and decreases the reference count of B by one with the function `g_object_unref (B)` .
If two objects A and B refer to C, then the reference count of C is two.
If A no longer needs C, A discards the pointer to C and decreases the reference count in C by one.
Now the reference count of C is one.
In the same way, if B no longer needs C, B discards the pointer to C and decreases the reference count in C by one.
At this moment, no object refers to C and the reference count of C is zero.
This means C is no longer useful.
Then C destructs itself and finally the memories allocated to C is freed.
![Reference count of B ](../image/refcount.png )
The idea above is based on an assumption that an object referred by nothing has reference count of zero.
When the reference count drops to zero, the object starts its destruction process.
The destruction process is split into two phases: disposing and finalizing.
2022-12-21 14:31:33 +01:00
In the disposing process, the object invokes the function pointed by `dispose` in its class to release all references to other instances.
After that, it invokes the function pointed by `finalize` in its class to complete the destruction process.
2021-06-15 04:24:42 +02:00
For example, dispose handler or dispose method.
2022-12-21 14:31:33 +01:00
In the destruction process, TfeTextView needs to unref the GFile pointed by `tv->file` .
You must write the dispose handler `tfe_text_view_dispose` .
2021-06-15 04:24:42 +02:00
~~~C
1 static void
2 tfe_text_view_dispose (GObject *gobject) {
3 TfeTextView *tv = TFE_TEXT_VIEW (gobject);
4
5 if (G_IS_FILE (tv->file))
6 g_clear_object (&tv->file);
7
8 G_OBJECT_CLASS (tfe_text_view_parent_class)->dispose (gobject);
9 }
~~~
- 5,6: If `tv->file` points a GFile, decrease its reference count.
`g_clear_object` decreases the reference count and assigns NULL to `tv->file` .
In dispose handlers, we usually use `g_clear_object` rather than `g_object_unref` .
- 8: invokes parent's dispose handler. (This will be explained later.)
In the disposing process, the object uses the pointer in its class to call the handler.
Therefore, `tfe_text_view_dispose` needs to be registered in the class when the TfeTextView class is initialized.
2022-12-21 14:31:33 +01:00
The function `tfe_text_view_class_init` is the class initialization function and it is declared in the `G_DEFINE_TYPE` macro expansion.
2021-06-15 04:24:42 +02:00
~~~C
static void
tfe_text_view_class_init (TfeTextViewClass *class) {
GObjectClass *object_class = G_OBJECT_CLASS (class);
object_class->dispose = tfe_text_view_dispose;
2021-01-13 05:29:35 +01:00
2021-06-15 04:24:42 +02:00
}
2021-01-25 13:20:15 +01:00
~~~
2021-01-13 05:29:35 +01:00
2021-06-17 13:04:16 +02:00
Each ancestors' class has been created before TfeTextViewClass is created.
2021-06-15 04:24:42 +02:00
Therefore, there are four classes and each class has a pointer to each dispose handler.
Look at the following diagram.
There are four classes -- GObjectClass (GInitiallyUnownedClass), GtkWidgetClass, GtkTextViewClass and TfeTextViewClass.
Each class has its own dispose handler -- `dh1` , `dh2` , `dh3` and `tfe_text_view_dispose` .
2021-01-13 05:29:35 +01:00
2021-06-15 04:24:42 +02:00
![dispose handlers ](../image/dispose_handler.png )
2021-01-13 05:29:35 +01:00
2021-06-15 04:24:42 +02:00
Now, look at the `tfe_text_view_dispose` program above.
It first releases the reference to GFile object pointed by `tv->file` .
Then it invokes its parent's dispose handler in line 8.
2021-01-13 05:29:35 +01:00
2021-01-25 13:20:15 +01:00
~~~C
2021-06-15 04:24:42 +02:00
G_OBJECT_CLASS (tfe_text_view_parent_class)->dispose (gobject);
2021-01-25 13:20:15 +01:00
~~~
2021-01-13 05:29:35 +01:00
2022-12-21 14:31:33 +01:00
A variable `tfe_text_view_parent_class` , which is made by `G_DEFINE_TYPE` macro, is a pointer that points the parent object class.
The variable `gobject` is a pointer to TfeTextView instance which is casted as a GObject instance.
2021-06-15 04:24:42 +02:00
Therefore, `G_OBJECT_CLASS (tfe_text_view_parent_class)->dispose` points the handler `dh3` in the diagram above.
2022-12-21 14:31:33 +01:00
The statement `G_OBJECT_CLASS (tfe_text_view_parent_class)->dispose (gobject)` is the same as `dh3 (gobject)` , which means it releases all the reference to the other instances in the GtkTextViewPrivate in the TfeTextView instance.
2021-06-15 04:24:42 +02:00
After that, `dh3` calls `dh2` , and `dh2` calls `dh1` .
Finally all the references are released.
2021-01-11 03:15:04 +01:00
2022-12-21 14:31:33 +01:00
Up: [README.md ](../README.md ), Prev: [Section 10 ](sec10.md ), Next: [Section 12 ](sec12.md )