mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-03 19:36:16 +01:00
3d55be0b82
Helps: #3037
1289 lines
41 KiB
Markdown
1289 lines
41 KiB
Markdown
Title: GObject Tutorial
|
||
|
||
# GObject Tutorial
|
||
|
||
## How to define and implement a new GObject
|
||
|
||
This document focuses on the implementation of a subtype of GObject, for
|
||
example to create a custom class hierarchy, or to subclass a GTK widget.
|
||
|
||
Throughout the chapter, a running example of a file viewer program is used,
|
||
which has a `ViewerFile` class to represent a single file being viewed, and
|
||
various derived classes for different types of files with special
|
||
functionality, such as audio files. The example application also supports
|
||
editing files (for example, to tweak a photo being viewed), using a
|
||
`ViewerEditable` interface.
|
||
|
||
### Boilerplate header code
|
||
|
||
The first step before writing the code for your GObject is to write the
|
||
type's header which contains the needed type, function and macro
|
||
definitions. Each of these elements is nothing but a convention which is
|
||
followed by almost all users of GObject, and has been refined over multiple
|
||
years of experience developing GObject-based code. If you are writing a
|
||
library, it is particularly important for you to adhere closely to these
|
||
conventions; users of your library will assume that you have. Even if not
|
||
writing a library, it will help other people who want to work on your
|
||
project.
|
||
|
||
Pick a name convention for your headers and source code and stick to it:
|
||
|
||
- use a dash to separate the prefix from the typename: `viewer-file.h` and
|
||
`viewer-file.c` (this is the convention used by most GNOME libraries and
|
||
applications)
|
||
- use an underscore to separate the prefix from the typename:
|
||
`viewer_file.h` and `viewer_file.c`
|
||
- do not separate the prefix from the typename: `viewerfile.h` and
|
||
`viewerfile.c` (this is the convention used by GTK)
|
||
|
||
Some people like the first two solutions better: it makes reading file names
|
||
easier for those with poor eyesight.
|
||
|
||
The basic conventions for any header which exposes a GType are described in
|
||
the section of the Type system introduction called
|
||
["Conventions"](concepts.html#conventions).
|
||
|
||
If you want to declare a type named "file" in the namespace "viewer", name
|
||
the type instance `ViewerFile` and its class `ViewerFileClass` (names are
|
||
case sensitive). The recommended method of declaring a type differs based on
|
||
whether the type is final or derivable.
|
||
|
||
Final types cannot be subclassed further, and should be the default choice
|
||
for new types—changing a final type to be derivable is always a change that
|
||
will be compatible with existing uses of the code, but the converse will
|
||
often cause problems. Final types are declared using the
|
||
`G_DECLARE_FINAL_TYPE` macro, and require a structure to hold the instance
|
||
data to be declared in the source code (not the header file).
|
||
|
||
```c
|
||
/*
|
||
* Copyright/Licensing information.
|
||
*/
|
||
|
||
/* inclusion guard */
|
||
#pragma once
|
||
|
||
#include <glib-object.h>
|
||
|
||
/*
|
||
* Potentially, include other headers on which this header depends.
|
||
*/
|
||
|
||
G_BEGIN_DECLS
|
||
|
||
/*
|
||
* Type declaration.
|
||
*/
|
||
#define VIEWER_TYPE_FILE viewer_file_get_type()
|
||
G_DECLARE_FINAL_TYPE (ViewerFile, viewer_file, VIEWER, FILE, GObject)
|
||
|
||
/*
|
||
* Method definitions.
|
||
*/
|
||
ViewerFile *viewer_file_new (void);
|
||
|
||
G_END_DECLS
|
||
```
|
||
|
||
Derivable types can be subclassed further, and their class and instance
|
||
structures form part of the public API which must not be changed if API
|
||
stability is cared about. They are declared using the
|
||
`G_DECLARE_DERIVABLE_TYPE` macro:
|
||
|
||
```c
|
||
/*
|
||
* Copyright/Licensing information.
|
||
*/
|
||
|
||
/* inclusion guard */
|
||
#pragma once
|
||
|
||
#include <glib-object.h>
|
||
|
||
/*
|
||
* Potentially, include other headers on which this header depends.
|
||
*/
|
||
|
||
G_BEGIN_DECLS
|
||
|
||
/*
|
||
* Type declaration.
|
||
*/
|
||
#define VIEWER_TYPE_FILE viewer_file_get_type()
|
||
G_DECLARE_DERIVABLE_TYPE (ViewerFile, viewer_file, VIEWER, FILE, GObject)
|
||
|
||
struct _ViewerFileClass
|
||
{
|
||
GObjectClass parent_class;
|
||
|
||
/* Class virtual function fields. */
|
||
void (* open) (ViewerFile *file,
|
||
GError **error);
|
||
|
||
/* Padding to allow adding up to 12 new virtual functions without
|
||
* breaking ABI. */
|
||
gpointer padding[12];
|
||
};
|
||
|
||
/*
|
||
* Method definitions.
|
||
*/
|
||
ViewerFile *viewer_file_new (void);
|
||
|
||
G_END_DECLS
|
||
```
|
||
|
||
The convention for header includes is to add the minimum number of
|
||
`#include` directives to the top of your headers needed to compile that
|
||
header. This allows client code to simply `#include "viewer-file.h"`,
|
||
without needing to know the prerequisites for `viewer-file.h`.
|
||
|
||
### Boilerplate code
|
||
|
||
In your code, the first step is to `#include` the needed headers:
|
||
|
||
```c
|
||
/*
|
||
* Copyright/Licensing information
|
||
*/
|
||
|
||
#include "viewer-file.h"
|
||
|
||
/* Private structure definition. */
|
||
typedef struct {
|
||
char *filename;
|
||
|
||
/* other private fields */
|
||
} ViewerFilePrivate;
|
||
|
||
/*
|
||
* forward definitions
|
||
*/
|
||
```
|
||
|
||
If the class is being declared as final using `G_DECLARE_FINAL_TYPE`, its instance structure should be defined in the C file:
|
||
|
||
```c
|
||
struct _ViewerFile
|
||
{
|
||
GObject parent_instance;
|
||
|
||
/* Other members, including private data. */
|
||
};
|
||
```
|
||
|
||
Call the `G_DEFINE_TYPE` macro (or `G_DEFINE_TYPE_WITH_PRIVATE` if your
|
||
class needs private data—final types do not need private data) using the
|
||
name of the type, the prefix of the functions and the parent GType to reduce
|
||
the amount of boilerplate needed. This macro will:
|
||
|
||
- implement the `viewer_file_get_type` function
|
||
- define a parent class pointer accessible from the whole `.c` file
|
||
- add private instance data to the type (if using `G_DEFINE_TYPE_WITH_PRIVATE`)
|
||
|
||
If the class has been declared as final using `G_DECLARE_FINAL_TYPE` private
|
||
data should be placed in the instance structure, `ViewerFile`, and
|
||
`G_DEFINE_TYPE` should be used instead of `G_DEFINE_TYPE_WITH_PRIVATE`. The
|
||
instance structure for a final class is not exposed publicly, and is not
|
||
embedded in the instance structures of any derived classes (because the
|
||
class is final); so its size can vary without causing incompatibilities for
|
||
code which uses the class. Conversely, private data for derivable classes
|
||
must be included in a private structure, and `G_DEFINE_TYPE_WITH_PRIVATE`
|
||
must be used.
|
||
|
||
```c
|
||
G_DEFINE_TYPE (ViewerFile, viewer_file, G_TYPE_OBJECT)
|
||
```
|
||
|
||
or
|
||
|
||
```c
|
||
G_DEFINE_TYPE_WITH_PRIVATE (ViewerFile, viewer_file, G_TYPE_OBJECT)
|
||
```
|
||
|
||
It is also possible to use the `G_DEFINE_TYPE_WITH_CODE` macro to control
|
||
the `get_type` function implementation — for instance, to add a call to the
|
||
`G_IMPLEMENT_INTERFACE` macro to implement an interface.
|
||
|
||
### Object construction
|
||
|
||
People often get confused when trying to construct their GObjects because of
|
||
the sheer number of different ways to hook into the objects's construction
|
||
process: it is difficult to figure which is the correct, recommended way.
|
||
|
||
The [documentation on object
|
||
instantiation](concepts.html#object-instantiation) shows what user-provided
|
||
functions are invoked during object instantiation and in which order they
|
||
are invoked. A user looking for the equivalent of the simple C++ constructor
|
||
function should use the `instance_init` method. It will be invoked after all
|
||
the parents’ `instance_init` functions have been invoked. It cannot take
|
||
arbitrary construction parameters (as in C++) but if your object needs
|
||
arbitrary parameters to complete initialization, you can use construction
|
||
properties.
|
||
|
||
Construction properties will be set only after all `instance_init` functions have run. No object reference will be returned to the client of `g_object_new()` until all the construction properties have been set.
|
||
|
||
It is important to note that object construction cannot ever fail. If you
|
||
require a fallible GObject construction, you can use the `GInitable` and
|
||
`GAsyncInitable` interfaces provided by the GIO library.
|
||
|
||
You should write the following code first:
|
||
|
||
```c
|
||
G_DEFINE_TYPE_WITH_PRIVATE (ViewerFile, viewer_file, G_TYPE_OBJECT)
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
}
|
||
|
||
static void
|
||
viewer_file_init (ViewerFile *self)
|
||
{
|
||
ViewerFilePrivate *priv = viewer_file_get_instance_private (self);
|
||
|
||
/* initialize all public and private members to reasonable default values.
|
||
* They are all automatically initialized to 0 to begin with. */
|
||
}
|
||
```
|
||
|
||
If you need special construction properties (with `G_PARAM_CONSTRUCT_ONLY`
|
||
set), install the properties in the `class_init()` function, override the
|
||
`set_property()` and `get_property()` methods of the GObject class, and
|
||
implement them as described by the section called ["Object
|
||
properties"](concepts.html#object-properties).
|
||
|
||
Property identifiers must start from 1, as 0 is reserved for internal use by GObject.
|
||
|
||
```c
|
||
enum
|
||
{
|
||
PROP_FILENAME = 1,
|
||
PROP_ZOOM_LEVEL,
|
||
N_PROPERTIES
|
||
};
|
||
|
||
static GParamSpec *obj_properties[N_PROPERTIES] = { NULL, };
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
||
object_class->set_property = viewer_file_set_property;
|
||
object_class->get_property = viewer_file_get_property;
|
||
|
||
obj_properties[PROP_FILENAME] =
|
||
g_param_spec_string ("filename",
|
||
"Filename",
|
||
"Name of the file to load and display from.",
|
||
NULL /* default value */,
|
||
G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE);
|
||
|
||
obj_properties[PROP_ZOOM_LEVEL] =
|
||
g_param_spec_uint ("zoom-level",
|
||
"Zoom level",
|
||
"Zoom level to view the file at.",
|
||
0 /* minimum value */,
|
||
10 /* maximum value */,
|
||
2 /* default value */,
|
||
G_PARAM_READWRITE);
|
||
|
||
g_object_class_install_properties (object_class,
|
||
N_PROPERTIES,
|
||
obj_properties);
|
||
}
|
||
```
|
||
|
||
If you need this, make sure you can build and run code similar to the code
|
||
shown above. Also, make sure your construct properties can be set without
|
||
side effects during construction.
|
||
|
||
Some people sometimes need to complete the initialization of an instance of
|
||
a type only after the properties passed to the constructors have been set.
|
||
This is possible through the use of the `constructor()` class method as
|
||
described in the section called "Object instantiation" or, more simply,
|
||
using the `constructed()` class method. Note that the `constructed()` virtual
|
||
function will only be invoked after the properties marked as
|
||
`G_PARAM_CONSTRUCT_ONLY` or `G_PARAM_CONSTRUCT` have been consumed, but before
|
||
the regular properties passed to `g_object_new()` have been set.
|
||
|
||
### Object destruction
|
||
|
||
Again, it is often difficult to figure out which mechanism to use to hook
|
||
into the object's destruction process: when the last `g_object_unref()` function
|
||
call is made, a lot of things happen as described in [the "Object memory
|
||
management" section](concepts.html#object-memory-management) of the
|
||
documentation.
|
||
|
||
The destruction process of your object is in two phases: dispose and
|
||
finalize. This split is necessary to handle potential cycles due to the
|
||
nature of the reference counting mechanism used by GObject, as well as
|
||
dealing with temporary revival of instances in case of signal emission
|
||
during the destruction sequence.
|
||
|
||
```c
|
||
struct _ViewerFilePrivate
|
||
{
|
||
gchar *filename;
|
||
guint zoom_level;
|
||
|
||
GInputStream *input_stream;
|
||
};
|
||
|
||
G_DEFINE_TYPE_WITH_PRIVATE (ViewerFile, viewer_file, G_TYPE_OBJECT)
|
||
|
||
static void
|
||
viewer_file_dispose (GObject *gobject)
|
||
{
|
||
ViewerFilePrivate *priv = viewer_file_get_instance_private (VIEWER_FILE (gobject));
|
||
|
||
/* In dispose(), you are supposed to free all types referenced from this
|
||
* object which might themselves hold a reference to self. Generally,
|
||
* the most simple solution is to unref all members on which you own a
|
||
* reference.
|
||
*/
|
||
|
||
/* dispose() might be called multiple times, so we must guard against
|
||
* calling g_object_unref() on an invalid GObject by setting the member
|
||
* NULL; g_clear_object() does this for us.
|
||
*/
|
||
g_clear_object (&priv->input_stream);
|
||
|
||
/* Always chain up to the parent class; there is no need to check if
|
||
* the parent class implements the dispose() virtual function: it is
|
||
* always guaranteed to do so
|
||
*/
|
||
G_OBJECT_CLASS (viewer_file_parent_class)->dispose (gobject);
|
||
}
|
||
|
||
static void
|
||
viewer_file_finalize (GObject *gobject)
|
||
{
|
||
ViewerFilePrivate *priv = viewer_file_get_instance_private (VIEWER_FILE (gobject));
|
||
|
||
g_free (priv->filename);
|
||
|
||
/* Always chain up to the parent class; as with dispose(), finalize()
|
||
* is guaranteed to exist on the parent's class virtual function table
|
||
*/
|
||
G_OBJECT_CLASS (viewer_file_parent_class)->finalize (gobject);
|
||
}
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
||
object_class->dispose = viewer_file_dispose;
|
||
object_class->finalize = viewer_file_finalize;
|
||
}
|
||
|
||
static void
|
||
viewer_file_init (ViewerFile *self);
|
||
{
|
||
ViewerFilePrivate *priv = viewer_file_get_instance_private (self);
|
||
|
||
priv->input_stream = g_object_new (VIEWER_TYPE_INPUT_STREAM, NULL);
|
||
priv->filename = /* would be set as a property */;
|
||
}
|
||
```
|
||
|
||
It is possible that object methods might be invoked after dispose is run and
|
||
before finalize runs. GObject does not consider this to be a program error:
|
||
you must gracefully detect this and neither crash nor warn the user, by
|
||
having a disposed instance revert to an inert state.
|
||
|
||
### Object methods
|
||
|
||
Just as with C++, there are many different ways to define object methods and
|
||
extend them: the following list and sections draw on C++ vocabulary.
|
||
(Readers are expected to know basic C++ concepts. Those who have not had to
|
||
write C++ code recently can refer to a [C++
|
||
tutorial](http://www.cplusplus.com/doc/tutorial/) to refresh their
|
||
memories.)
|
||
|
||
- non-virtual public methods,
|
||
- virtual public methods and
|
||
- virtual private methods
|
||
- non-virtual private methods
|
||
|
||
#### Non-Virtual Methods
|
||
|
||
These are the simplest, providing a simple method which acts on the object.
|
||
Provide a function prototype in the header and an implementation of that
|
||
prototype in the source file.
|
||
|
||
```c
|
||
/* declaration in the header. */
|
||
void viewer_file_open (ViewerFile *self,
|
||
GError **error);
|
||
```
|
||
|
||
```c
|
||
/* implementation in the source file */
|
||
void
|
||
viewer_file_open (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
/* do stuff here. */
|
||
}
|
||
```
|
||
|
||
#### Virtual public methods
|
||
|
||
This is the preferred way to create GObjects with overridable methods:
|
||
|
||
- define the common method and its virtual function in the class structure
|
||
in the public header
|
||
- define the common method in the header file and implement it in the source
|
||
file
|
||
- implement a base version of the virtual function in the source file and
|
||
initialize the virtual function pointer to this implementation in the
|
||
object’s `class_init` function; or leave it as `NULL` for a ‘pure virtual’
|
||
method which must be overridden by derived classes
|
||
- re-implement the virtual function in each derived class which needs to
|
||
override it
|
||
|
||
Note that virtual functions can only be defined if the class is derivable,
|
||
declared using `G_DECLARE_DERIVABLE_TYPE` so the class structure can be
|
||
defined.
|
||
|
||
```c
|
||
/* declaration in viewer-file.h. */
|
||
#define VIEWER_TYPE_FILE viewer_file_get_type ()
|
||
G_DECLARE_DERIVABLE_TYPE (ViewerFile, viewer_file, VIEWER, FILE, GObject)
|
||
|
||
struct _ViewerFileClass
|
||
{
|
||
GObjectClass parent_class;
|
||
|
||
/* stuff */
|
||
void (*open) (ViewerFile *self,
|
||
GError **error);
|
||
|
||
/* Padding to allow adding up to 12 new virtual functions without
|
||
* breaking ABI. */
|
||
gpointer padding[12];
|
||
};
|
||
|
||
void viewer_file_open (ViewerFile *self,
|
||
GError **error);
|
||
```
|
||
|
||
```c
|
||
/* implementation in viewer-file.c */
|
||
void
|
||
viewer_file_open (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
ViewerFileClass *klass;
|
||
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
klass = VIEWER_FILE_GET_CLASS (self);
|
||
g_return_if_fail (klass->open != NULL);
|
||
|
||
klass->open (self, error);
|
||
}
|
||
```
|
||
|
||
The code above simply redirects the open call to the relevant virtual
|
||
function.
|
||
|
||
It is possible to provide a default implementation for this class method in
|
||
the object's `class_init` function: initialize the `klass->open` field to a
|
||
pointer to the actual implementation. By default, class methods that are not
|
||
inherited are initialized to `NULL`, and thus are to be considered "pure
|
||
virtual".
|
||
|
||
```c
|
||
static void
|
||
viewer_file_real_close (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
/* Default implementation for the virtual method. */
|
||
}
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
/* this is not necessary, except for demonstration purposes.
|
||
*
|
||
* pure virtual method: mandates implementation in children.
|
||
*/
|
||
klass->open = NULL;
|
||
|
||
/* merely virtual method. */
|
||
klass->close = viewer_file_real_close;
|
||
}
|
||
|
||
void
|
||
viewer_file_open (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
ViewerFileClass *klass;
|
||
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
klass = VIEWER_FILE_GET_CLASS (self);
|
||
|
||
/* if the method is purely virtual, then it is a good idea to
|
||
* check that it has been overridden before calling it, and,
|
||
* depending on the intent of the class, either ignore it silently
|
||
* or warn the user.
|
||
*/
|
||
g_return_if_fail (klass->open != NULL);
|
||
klass->open (self, error);
|
||
}
|
||
|
||
void
|
||
viewer_file_close (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
ViewerFileClass *klass;
|
||
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
klass = VIEWER_FILE_GET_CLASS (self);
|
||
if (klass->close != NULL)
|
||
klass->close (self, error);
|
||
}
|
||
```
|
||
|
||
#### Virtual private Methods
|
||
|
||
These are very similar to virtual public methods. They just don't have a
|
||
public function to call directly. The header file contains only a
|
||
declaration of the virtual function:
|
||
|
||
```c
|
||
/* declaration in viewer-file.h. */
|
||
struct _ViewerFileClass
|
||
{
|
||
GObjectClass parent;
|
||
|
||
/* Public virtual method as before. */
|
||
void (*open) (ViewerFile *self,
|
||
GError **error);
|
||
|
||
/* Private helper function to work out whether the file can be loaded via
|
||
* memory mapped I/O, or whether it has to be read as a stream. */
|
||
gboolean (*can_memory_map) (ViewerFile *self);
|
||
|
||
/* Padding to allow adding up to 12 new virtual functions without
|
||
* breaking ABI. */
|
||
gpointer padding[12];
|
||
};
|
||
|
||
void viewer_file_open (ViewerFile *self, GError **error);
|
||
```
|
||
|
||
These virtual functions are often used to delegate part of the job to child classes:
|
||
|
||
```c
|
||
/* this accessor function is static: it is not exported outside of this file. */
|
||
static gboolean
|
||
viewer_file_can_memory_map (ViewerFile *self)
|
||
{
|
||
return VIEWER_FILE_GET_CLASS (self)->can_memory_map (self);
|
||
}
|
||
|
||
void
|
||
viewer_file_open (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
/*
|
||
* Try to load the file using memory mapped I/O, if the implementation of the
|
||
* class determines that is possible using its private virtual method.
|
||
*/
|
||
if (viewer_file_can_memory_map (self))
|
||
{
|
||
/* Load the file using memory mapped I/O. */
|
||
}
|
||
else
|
||
{
|
||
/* Fall back to trying to load the file using streaming I/O… */
|
||
}
|
||
}
|
||
```
|
||
|
||
Again, it is possible to provide a default implementation for this private virtual function:
|
||
|
||
```c
|
||
static gboolean
|
||
viewer_file_real_can_memory_map (ViewerFile *self)
|
||
{
|
||
/* As an example, always return false. Or, potentially return true if the
|
||
* file is local. */
|
||
return FALSE;
|
||
}
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
/* non-pure virtual method; does not have to be implemented in children. */
|
||
klass->can_memory_map = viewer_file_real_can_memory_map;
|
||
}
|
||
```
|
||
|
||
Derived classes can then override the method with code such as:
|
||
|
||
```c
|
||
static void
|
||
viewer_audio_file_class_init (ViewerAudioFileClass *klass)
|
||
{
|
||
ViewerFileClass *file_class = VIEWER_FILE_CLASS (klass);
|
||
|
||
/* implement pure virtual function. */
|
||
file_class->can_memory_map = viewer_audio_file_can_memory_map;
|
||
}
|
||
```
|
||
|
||
### Chaining up
|
||
|
||
Chaining up is often loosely defined by the following set of conditions:
|
||
|
||
- parent class A defines a public virtual method named `foo` and provides a
|
||
default implementation
|
||
- child class B re-implements method `foo`
|
||
- B’s implementation of `foo` calls (‘chains up to’) its parent class A’s
|
||
implementation of `foo`
|
||
|
||
There are various uses of this idiom:
|
||
|
||
- you need to extend the behaviour of a class without modifying its code.
|
||
You create a subclass to inherit its implementation, re-implement a public
|
||
virtual method to modify the behaviour and chain up to ensure that the
|
||
previous behaviour is not really modified, just extended
|
||
- you need to implement the
|
||
[Chain of Responsibility pattern](https://en.wikipedia.org/wiki/Chain-of-responsibility_pattern):
|
||
each object of the inheritance tree chains up to its parent (typically, at the
|
||
beginning or the end of the method) to ensure that each handler is run in turn
|
||
|
||
To explicitly chain up to the implementation of the virtual method in the
|
||
parent class, you first need a handle to the original parent class
|
||
structure. This pointer can then be used to access the original virtual
|
||
function pointer and invoke it directly
|
||
|
||
The "original" adjective used in the sentence above is not innocuous. To
|
||
fully understand its meaning, recall how class structures are initialized:
|
||
for each object type, the class structure associated with this object is
|
||
created by first copying the class structure of its parent type (a simple
|
||
memcpy) and then by invoking the `class_init` callback on the resulting class
|
||
structure. Since the `class_init` callback is responsible for overwriting the
|
||
class structure with the user re-implementations of the class methods, the
|
||
modified copy of the parent class structure stored in the derived instance
|
||
cannot be used. A copy of the class structure of an instance of the parent
|
||
class is needed.
|
||
|
||
To chain up, you can use the `parent_class` pointer created and initialized
|
||
by the `G_DEFINE_TYPE` family of macros, for instance:
|
||
|
||
```c
|
||
static void
|
||
b_method_to_call (B *obj, int some_param)
|
||
{
|
||
/* do stuff before chain up */
|
||
|
||
/* call the method_to_call() virtual function on the
|
||
* parent of BClass, AClass.
|
||
*
|
||
* remember the explicit cast to AClass*
|
||
*/
|
||
A_CLASS (b_parent_class)->method_to_call (obj, some_param);
|
||
|
||
/* do stuff after chain up */
|
||
}
|
||
```
|
||
|
||
## How to define and implement interfaces
|
||
|
||
### Defining interfaces
|
||
|
||
The theory behind how GObject interfaces work is given in the section called
|
||
["Non-instantiatable classed types:
|
||
interfaces"](concepts.html#non-instantiatable-classed-types-interfaces);
|
||
this section covers how to define and implement an interface.
|
||
|
||
The first step is to get the header right. This interface defines three
|
||
methods:
|
||
|
||
```c
|
||
/*
|
||
* Copyright/Licensing information.
|
||
*/
|
||
|
||
#pragma once
|
||
|
||
#include <glib-object.h>
|
||
|
||
G_BEGIN_DECLS
|
||
|
||
#define VIEWER_TYPE_EDITABLE viewer_editable_get_type()
|
||
G_DECLARE_INTERFACE (ViewerEditable, viewer_editable, VIEWER, EDITABLE, GObject)
|
||
|
||
struct _ViewerEditableInterface
|
||
{
|
||
GTypeInterface parent_iface;
|
||
|
||
void (*save) (ViewerEditable *self,
|
||
GError **error);
|
||
void (*undo) (ViewerEditable *self,
|
||
guint n_steps);
|
||
void (*redo) (ViewerEditable *self,
|
||
guint n_steps);
|
||
};
|
||
|
||
void viewer_editable_save (ViewerEditable *self,
|
||
GError **error);
|
||
void viewer_editable_undo (ViewerEditable *self,
|
||
guint n_steps);
|
||
void viewer_editable_redo (ViewerEditable *self,
|
||
guint n_steps);
|
||
|
||
G_END_DECLS
|
||
```
|
||
|
||
This code is the same as the code for a normal GType which derives from a
|
||
GObject except for a few details:
|
||
|
||
- the `_GET_CLASS` function is called `_GET_IFACE` (and is defined by `G_DECLARE_INTERFACE`)
|
||
- the instance type, `ViewerEditable`, is not fully defined: it is used merely as an abstract type which represents an instance of whatever object which implements the interface
|
||
- the parent of the `ViewerEditableInterface` is `GTypeInterface`, not `GObjectClass`
|
||
|
||
The implementation of the `ViewerEditable` type itself is trivial:
|
||
|
||
- `G_DEFINE_INTERFACE` creates a `viewer_editable_get_type` function which registers the type in the type system. The third argument is used to define a prerequisite interface (which we'll talk about more later). Just pass 0 for this argument when an interface has no prerequisite
|
||
- `viewer_editable_default_init` is expected to register the interface's signals if there are any (we will see a bit later how to use them)
|
||
- the interface methods `viewer_editable_save`, `viewer_editable_undo` and `viewer_editable_redo` dereference the interface structure to access its associated interface function and call it
|
||
|
||
```c
|
||
G_DEFINE_INTERFACE (ViewerEditable, viewer_editable, G_TYPE_OBJECT)
|
||
|
||
static void
|
||
viewer_editable_default_init (ViewerEditableInterface *iface)
|
||
{
|
||
/* add properties and signals to the interface here */
|
||
}
|
||
|
||
void
|
||
viewer_editable_save (ViewerEditable *self,
|
||
GError **error)
|
||
{
|
||
ViewerEditableInterface *iface;
|
||
|
||
g_return_if_fail (VIEWER_IS_EDITABLE (self));
|
||
g_return_if_fail (error == NULL || *error == NULL);
|
||
|
||
iface = VIEWER_EDITABLE_GET_IFACE (self);
|
||
g_return_if_fail (iface->save != NULL);
|
||
iface->save (self, error);
|
||
}
|
||
|
||
void
|
||
viewer_editable_undo (ViewerEditable *self,
|
||
guint n_steps)
|
||
{
|
||
ViewerEditableInterface *iface;
|
||
|
||
g_return_if_fail (VIEWER_IS_EDITABLE (self));
|
||
|
||
iface = VIEWER_EDITABLE_GET_IFACE (self);
|
||
g_return_if_fail (iface->undo != NULL);
|
||
iface->undo (self, n_steps);
|
||
}
|
||
|
||
void
|
||
viewer_editable_redo (ViewerEditable *self,
|
||
guint n_steps)
|
||
{
|
||
ViewerEditableInterface *iface;
|
||
|
||
g_return_if_fail (VIEWER_IS_EDITABLE (self));
|
||
|
||
iface = VIEWER_EDITABLE_GET_IFACE (self);
|
||
g_return_if_fail (iface->redo != NULL);
|
||
iface->redo (self, n_steps);
|
||
}
|
||
```
|
||
|
||
### Implementing interfaces
|
||
|
||
Once the interface is defined, implementing it is rather trivial.
|
||
|
||
The first step is to define a normal final GObject class exactly as usual.
|
||
|
||
The second step is to implement `ViewerFile` by defining it using
|
||
`G_DEFINE_TYPE_WITH_CODE` and `G_IMPLEMENT_INTERFACE` instead of
|
||
`G_DEFINE_TYPE`:
|
||
|
||
```c
|
||
static void viewer_file_editable_interface_init (ViewerEditableInterface *iface);
|
||
|
||
G_DEFINE_TYPE_WITH_CODE (ViewerFile, viewer_file, G_TYPE_OBJECT,
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE,
|
||
viewer_file_editable_interface_init))
|
||
```
|
||
|
||
This definition is very much like all the similar functions seen previously.
|
||
The only interface-specific code present here is the use of
|
||
`G_IMPLEMENT_INTERFACE`.
|
||
|
||
Classes can implement multiple interfaces by using multiple calls to
|
||
`G_IMPLEMENT_INTERFACE` inside the call to `G_DEFINE_TYPE_WITH_CODE`.
|
||
|
||
`viewer_file_editable_interface_init` is the interface initialization
|
||
function: inside it, every virtual method of the interface must be assigned
|
||
to its implementation:
|
||
|
||
```c
|
||
static void
|
||
viewer_file_editable_save (ViewerFile *self,
|
||
GError **error)
|
||
{
|
||
g_print ("File implementation of editable interface save method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_undo (ViewerFile *self,
|
||
guint n_steps)
|
||
{
|
||
g_print ("File implementation of editable interface undo method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_redo (ViewerFile *self,
|
||
guint n_steps)
|
||
{
|
||
g_print ("File implementation of editable interface redo method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_interface_init (ViewerEditableInterface *iface)
|
||
{
|
||
iface->save = viewer_file_editable_save;
|
||
iface->undo = viewer_file_editable_undo;
|
||
iface->redo = viewer_file_editable_redo;
|
||
}
|
||
|
||
static void
|
||
viewer_file_init (ViewerFile *self)
|
||
{
|
||
/* Instance variable initialisation code. */
|
||
}
|
||
```
|
||
|
||
If the object is not of final type, e.g. was declared using
|
||
`G_DECLARE_DERIVABLE_TYPE` then `G_ADD_PRIVATE` macro should be added. The
|
||
private structure should be declared exactly as for a normal derivable
|
||
object.
|
||
|
||
```c
|
||
G_DEFINE_TYPE_WITH_CODE (ViewerFile, viewer_file, G_TYPE_OBJECT,
|
||
G_ADD_PRIVATE (ViewerFile)
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE,
|
||
viewer_file_editable_interface_init))
|
||
```
|
||
|
||
### Interface definition prerequisites
|
||
|
||
To specify that an interface requires the presence of other interfaces when
|
||
implemented, GObject introduces the concept of prerequisites: it is possible
|
||
to associate a list of prerequisite types to an interface. For example, if
|
||
object A wishes to implement interface I1, and if interface I1 has a
|
||
prerequisite on interface I2, A has to implement both I1 and I2.
|
||
|
||
The mechanism described above is, in practice, very similar to Java's
|
||
interface I1 extends interface I2. The example below shows the GObject
|
||
equivalent:
|
||
|
||
```
|
||
/* Make the ViewerEditableLossy interface require ViewerEditable interface. */
|
||
G_DEFINE_INTERFACE (ViewerEditableLossy, viewer_editable_lossy, VIEWER_TYPE_EDITABLE)
|
||
```
|
||
|
||
In the `G_DEFINE_INTERFACE` call above, the third parameter defines the
|
||
prerequisite type. This is the GType of either an interface or a class. In
|
||
this case the `ViewerEditable` interface is a prerequisite of
|
||
`ViewerEditableLossy`. The code below shows how an implementation can
|
||
implement both interfaces and register their implementations:
|
||
|
||
```c
|
||
static void
|
||
viewer_file_editable_lossy_compress (ViewerEditableLossy *editable)
|
||
{
|
||
ViewerFile *self = VIEWER_FILE (editable);
|
||
|
||
g_print ("File implementation of lossy editable interface compress method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_lossy_interface_init (ViewerEditableLossyInterface *iface)
|
||
{
|
||
iface->compress = viewer_file_editable_lossy_compress;
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_save (ViewerEditable *editable,
|
||
GError **error)
|
||
{
|
||
ViewerFile *self = VIEWER_FILE (editable);
|
||
|
||
g_print ("File implementation of editable interface save method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_undo (ViewerEditable *editable,
|
||
guint n_steps)
|
||
{
|
||
ViewerFile *self = VIEWER_FILE (editable);
|
||
|
||
g_print ("File implementation of editable interface undo method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_redo (ViewerEditable *editable,
|
||
guint n_steps)
|
||
{
|
||
ViewerFile *self = VIEWER_FILE (editable);
|
||
|
||
g_print ("File implementation of editable interface redo method: %s.\n",
|
||
self->filename);
|
||
}
|
||
|
||
static void
|
||
viewer_file_editable_interface_init (ViewerEditableInterface *iface)
|
||
{
|
||
iface->save = viewer_file_editable_save;
|
||
iface->undo = viewer_file_editable_undo;
|
||
iface->redo = viewer_file_editable_redo;
|
||
}
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
/* Nothing here. */
|
||
}
|
||
|
||
static void
|
||
viewer_file_init (ViewerFile *self)
|
||
{
|
||
/* Instance variable initialisation code. */
|
||
}
|
||
|
||
G_DEFINE_TYPE_WITH_CODE (ViewerFile, viewer_file, G_TYPE_OBJECT,
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE,
|
||
viewer_file_editable_interface_init)
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE_LOSSY,
|
||
viewer_file_editable_lossy_interface_init))
|
||
```
|
||
|
||
It is very important to notice that the order in which interface
|
||
implementations are added to the main object is not random:
|
||
`g_type_add_interface_static()`, which is called by `G_IMPLEMENT_INTERFACE`, must
|
||
be invoked first on the interfaces which have no prerequisites and then on
|
||
the others.
|
||
|
||
### Interface properties
|
||
|
||
GObject interfaces can also have properties. Declaration of the interface
|
||
properties is similar to declaring the properties of ordinary GObject types
|
||
as explained in the section called ["Object
|
||
properties"](concepts.html#object-properties), except that
|
||
`g_object_interface_install_property()` is used to declare the properties
|
||
instead of `g_object_class_install_property()`.
|
||
|
||
To include a property named 'autosave-frequency' of type gdouble in the
|
||
`ViewerEditable` interface example code above, we only need to add one call in
|
||
`viewer_editable_default_init()` as shown below:
|
||
|
||
```c
|
||
static void
|
||
viewer_editable_default_init (ViewerEditableInterface *iface)
|
||
{
|
||
g_object_interface_install_property (iface,
|
||
g_param_spec_double ("autosave-frequency",
|
||
"Autosave frequency",
|
||
"Frequency (in per-seconds) to autosave backups of the editable content at. "
|
||
"Or zero to disable autosaves.",
|
||
0.0, /* minimum */
|
||
G_MAXDOUBLE, /* maximum */
|
||
0.0, /* default */
|
||
G_PARAM_READWRITE));
|
||
}
|
||
```
|
||
|
||
One point worth noting is that the declared property wasn't assigned an
|
||
integer ID. The reason being that integer IDs of properties are used only
|
||
inside the `get_property` and `set_property` virtual methods. Since interfaces
|
||
declare but do not implement properties, there is no need to assign integer
|
||
IDs to them.
|
||
|
||
An implementation declares and defines its properties in the usual way as
|
||
explained in the section called “Object properties”, except for one small
|
||
change: it can declare the properties of the interface it implements using
|
||
`g_object_class_override_property()` instead of `g_object_class_install_property()`.
|
||
The following code snippet shows the modifications needed in the `ViewerFile`
|
||
declaration and implementation above:
|
||
|
||
```c
|
||
struct _ViewerFile
|
||
{
|
||
GObject parent_instance;
|
||
|
||
double autosave_frequency;
|
||
};
|
||
|
||
enum
|
||
{
|
||
PROP_AUTOSAVE_FREQUENCY = 1,
|
||
N_PROPERTIES
|
||
};
|
||
|
||
static void
|
||
viewer_file_set_property (GObject *object,
|
||
guint prop_id,
|
||
const GValue *value,
|
||
GParamSpec *pspec)
|
||
{
|
||
ViewerFile *file = VIEWER_FILE (object);
|
||
|
||
switch (prop_id)
|
||
{
|
||
case PROP_AUTOSAVE_FREQUENCY:
|
||
file->autosave_frequency = g_value_get_double (value);
|
||
break;
|
||
|
||
default:
|
||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
break;
|
||
}
|
||
}
|
||
|
||
static void
|
||
viewer_file_get_property (GObject *object,
|
||
guint prop_id,
|
||
GValue *value,
|
||
GParamSpec *pspec)
|
||
{
|
||
ViewerFile *file = VIEWER_FILE (object);
|
||
|
||
switch (prop_id)
|
||
{
|
||
case PROP_AUTOSAVE_FREQUENCY:
|
||
g_value_set_double (value, file->autosave_frequency);
|
||
break;
|
||
|
||
default:
|
||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
break;
|
||
}
|
||
}
|
||
|
||
static void
|
||
viewer_file_class_init (ViewerFileClass *klass)
|
||
{
|
||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
|
||
object_class->set_property = viewer_file_set_property;
|
||
object_class->get_property = viewer_file_get_property;
|
||
|
||
g_object_class_override_property (object_class, PROP_AUTOSAVE_FREQUENCY, "autosave-frequency");
|
||
}
|
||
```
|
||
|
||
### Overriding interface methods
|
||
|
||
If a base class already implements an interface and a derived class needs to
|
||
implement the same interface but needs to override certain methods, you must
|
||
reimplement the interface and set only the interface methods which need
|
||
overriding.
|
||
|
||
In this example, `ViewerAudioFile` is derived from `ViewerFile`. Both implement
|
||
the `ViewerEditable` interface. `ViewerAudioFile` only implements one method of
|
||
the `ViewerEditable` interface and uses the base class implementation of the
|
||
other.
|
||
|
||
```c
|
||
static void
|
||
viewer_audio_file_editable_save (ViewerEditable *editable,
|
||
GError **error)
|
||
{
|
||
ViewerAudioFile *self = VIEWER_AUDIO_FILE (editable);
|
||
|
||
g_print ("Audio file implementation of editable interface save method.\n");
|
||
}
|
||
|
||
static void
|
||
viewer_audio_file_editable_interface_init (ViewerEditableInterface *iface)
|
||
{
|
||
/* Override the implementation of save(). */
|
||
iface->save = viewer_audio_file_editable_save;
|
||
|
||
/*
|
||
* Leave iface->undo and ->redo alone, they are already set to the
|
||
* base class implementation.
|
||
*/
|
||
}
|
||
|
||
G_DEFINE_TYPE_WITH_CODE (ViewerAudioFile, viewer_audio_file, VIEWER_TYPE_FILE,
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE,
|
||
viewer_audio_file_editable_interface_init))
|
||
|
||
static void
|
||
viewer_audio_file_class_init (ViewerAudioFileClass *klass)
|
||
{
|
||
/* Nothing here. */
|
||
}
|
||
|
||
static void
|
||
viewer_audio_file_init (ViewerAudioFile *self)
|
||
{
|
||
/* Nothing here. */
|
||
}
|
||
```
|
||
|
||
To access the base class interface implementation use
|
||
`g_type_interface_peek_parent()` from within an interface's `default_init`
|
||
function.
|
||
|
||
To call the base class implementation of an interface method from a derived
|
||
class where than interface method has been overridden, stash away the
|
||
pointer returned from `g_type_interface_peek_parent()` in a global variable.
|
||
|
||
In this example `ViewerAudioFile` overrides the save interface method. In
|
||
its overridden method it calls the base class implementation of the same
|
||
interface method.
|
||
|
||
```c
|
||
static ViewerEditableInterface *viewer_editable_parent_interface = NULL;
|
||
|
||
static void
|
||
viewer_audio_file_editable_save (ViewerEditable *editable,
|
||
GError **error)
|
||
{
|
||
ViewerAudioFile *self = VIEWER_AUDIO_FILE (editable);
|
||
|
||
g_print ("Audio file implementation of editable interface save method.\n");
|
||
|
||
/* Now call the base implementation */
|
||
viewer_editable_parent_interface->save (editable, error);
|
||
}
|
||
|
||
static void
|
||
viewer_audio_file_editable_interface_init (ViewerEditableInterface *iface)
|
||
{
|
||
viewer_editable_parent_interface = g_type_interface_peek_parent (iface);
|
||
|
||
iface->save = viewer_audio_file_editable_save;
|
||
}
|
||
|
||
G_DEFINE_TYPE_WITH_CODE (ViewerAudioFile, viewer_audio_file, VIEWER_TYPE_FILE,
|
||
G_IMPLEMENT_INTERFACE (VIEWER_TYPE_EDITABLE,
|
||
viewer_audio_file_editable_interface_init))
|
||
|
||
static void
|
||
viewer_audio_file_class_init (ViewerAudioFileClass *klass)
|
||
{
|
||
/* Nothing here. */
|
||
}
|
||
|
||
static void
|
||
viewer_audio_file_init (ViewerAudioFile *self)
|
||
{
|
||
/* Nothing here. */
|
||
}
|
||
```
|
||
|
||
## How to create and use signals
|
||
|
||
The signal system in GType is pretty complex and flexible: it is possible
|
||
for its users to connect at runtime any number of callbacks (implemented in
|
||
any language for which a binding exists) to any signal and to stop the
|
||
emission of any signal at any state of the signal emission process. This
|
||
flexibility makes it possible to use GSignal for much more than just
|
||
emitting signals to multiple clients.
|
||
|
||
### Simple use of signals
|
||
|
||
The most basic use of signals is to implement event notification. For
|
||
example, given a `ViewerFile` object with a write method, a signal could be
|
||
emitted whenever the file is changed using that method. The code below shows
|
||
how the user can connect a callback to the "changed" signal.
|
||
|
||
```c
|
||
file = g_object_new (VIEWER_FILE_TYPE, NULL);
|
||
|
||
g_signal_connect (file, "changed", (GCallback) changed_event, NULL);
|
||
|
||
viewer_file_write (file, buffer, strlen (buffer));
|
||
```
|
||
|
||
The ViewerFile signal is registered in the `class_init` function:
|
||
|
||
```c
|
||
file_signals[CHANGED] =
|
||
g_signal_newv ("changed",
|
||
G_TYPE_FROM_CLASS (object_class),
|
||
G_SIGNAL_RUN_LAST | G_SIGNAL_NO_RECURSE | G_SIGNAL_NO_HOOKS,
|
||
NULL /* closure */,
|
||
NULL /* accumulator */,
|
||
NULL /* accumulator data */,
|
||
NULL /* C marshaller */,
|
||
G_TYPE_NONE /* return_type */,
|
||
0 /* n_params */,
|
||
NULL /* param_types */);
|
||
```
|
||
|
||
and the signal is emitted in `viewer_file_write`:
|
||
|
||
```c
|
||
void
|
||
viewer_file_write (ViewerFile *self,
|
||
const guint8 *buffer,
|
||
gsize size)
|
||
{
|
||
g_return_if_fail (VIEWER_IS_FILE (self));
|
||
g_return_if_fail (buffer != NULL || size == 0);
|
||
|
||
/* First write data. */
|
||
|
||
/* Then, notify user of data written. */
|
||
g_signal_emit (self, file_signals[CHANGED], 0 /* details */);
|
||
}
|
||
```
|
||
|
||
As shown above, the details parameter can safely be set to zero if no detail
|
||
needs to be conveyed. For a discussion of what it can be used for, see the
|
||
section called [“The detail argument”](concepts.html#the-detail-argument).
|
||
|
||
The C signal marshaller should always be `NULL`, in which case the best
|
||
marshaller for the given closure type will be chosen by GLib. This may be an
|
||
internal marshaller specific to the closure type, or
|
||
`g_cclosure_marshal_generic()`, which implements generic conversion of arrays of
|
||
parameters to C callback invocations. GLib used to require the user to write
|
||
or generate a type-specific marshaller and pass that, but that has been
|
||
deprecated in favour of automatic selection of marshallers.
|
||
|
||
Note that `g_cclosure_marshal_generic()` is slower than non-generic
|
||
marshallers, so should be avoided for performance critical code. However,
|
||
performance critical code should rarely be using signals anyway, as signals
|
||
are synchronous, and the emission blocks until all listeners are invoked,
|
||
which has potentially unbounded cost.
|