GLib/GTK+ Development Guide
GLib/GTK+ Development Guide
A Getting Started
Version 0.8
Sébastien Wilmet
Content
1 Introductio 3
1.1 License . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3
1.2 Financial . . . . . . . . . . . . . . . . . . . . . . . . . . 3
1.3 Todo List for this Book and a Quick 2019 . . . . . . . . 4
1.4 What is GLib and . . . . . . . . . . . . . . . . . . . . . 4
1.5 The GNOME Desktop . . . . . . . . . . . . . . . . . . . . . . . 5
1.6 Prerequisites . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
1.7 Why and When Using the C . . . . . . . . . . . . . . 7
1.7.1 Separate the Backend from the Frontend . . . . . . . . . 7
1.7.2 Other Aspects to Keep in . . . . . . . . . . . . . . 8
1.8 Learning . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
1.9 The Development Environment . . . . . . . . . . . . . . . . . . . 10
1.10 Acknowledgments . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
3.1.6 #includ . . . . . . . . . . . . . . . . . . . . . . . . . . . 39
3.1.7 Type . . . . . . . . . . . . . . . . . . . . . . . 40
3.1.8 Object . . . . . . . . . . . . . . . . . . . . . 40
3.1.9 Object Destructor . . . . . . . . . . . . . . . . . . . . . . 40
3.1.10 Other Public Functions . . . . . . . . . . . . . . . . . . . 41
3.2 The Corresponding *.c File . . . . . . . . . . . . . . . . . . . . . 41
3.2.1 Order of . . . . . . . . . . . . . . . . . . . . . 43
3.2.2 GTK-Doc Comments . . . . . . . . . . . . . . . . . . . . 43
3.2.3 GObject Introspection Annotations .. . . . . . . . . . . . 44
3.2.4 Static vs Non-Static . . . . . . . . . . . . . . . 44
3.2.5 Defensive Programming . . . . . . . . . . . . . .
Chapter
Introductio
1. Licens
1. Financial
3
1. Todo List for this Book and a Quick 2019
• Make chapter 5 (about GTK) easier to understand and adapt the rest of
the document to better integrate the chapter;
• GTK+ has been renamed to GTK, the plus sign has been Adap
the text accordingly;
• No longer recommend Jhbuild, recommend instead Flatpak1 and
Stream2 ;
• Talk about the freedesktop.org3 specifications;
• The majority of active GNOME modules have been migrated to the Meson
build system now (instead of the Autotools);
• Write a chapter about developing C/GObject libraries;
• Talk about the Rust programming language.
4
The first version of GTK or the GIMP Ki 4 , was mainly written by
Peter Mattis in 1996 for the GIMP (GNU Image Manipulation but
has quickly become a general-purpose library. The “+” has been added
to distinguish between the original version and a new version that added object
oriented features. GLib started as part GTK but is now a standalone
library.
The GLib and GTK+ APIs are documented with GTK- Special comments
are written in the source code,and GTK-Doc extracts those comments to gen-
erate HTML
Although GLib and GTK+ are written in language bindings are available
for Python Perl and many other programming languages. At
beginning, manual bindings were created, which needed to be updated each time
the API the library changed. Nowadays the language bindings are generic
and are thus automatically updated when, for example, new functions are added.
This is thanks to GObject Special annotations are added to the
GTK-Doc to expose more information than what the C syntax can
5
provide, for example about ownership transfer of dynamically-allocated content
.
Any C library with GObject Introspection support is thus available from many
programming languages. In addition, the annotations are also useful to the
programmer because it’s a good and succint way to document certain recurrent
API
GLib and GTK+ are part of the GNU Project, whose overall goal is developing
free operating system (named GNU) plus applications to go withGNU
it.
for “GNU’s Not Unix”, a humorous way of saying that the GNU operating
is Unix-compatible. You can learn more about GNU at
The GLib/GTK+ web site http://
www.gtk.org
1. The GNOME
5
desktop environment and application suite.From the programmer’s perspective,
it is an application development framework (made up of numerous useful libraries
that are based on GLib and Applications written with the
libraries run fine even if the user isn’t running the desktop environment, but
they integrate nicely with the GNOME desktop if it’s
The desktop environment includes a “shell” for task switching and launching
programs, a “control center” for configuration, lots of applications such as a file
manager, a web browser, a movie player, etc.
These programs hide the traditional
Unix command line behind an easy-to-use graphical interface.
GNOME’s development framework makes it possible to write consistent, easy-to-
use, interoperable applications.The designers of windowing systems such as X11
or Wayland made a deliberate decision not to impose any user interface policy on
developers; GNOME adds a “policy layer,” creating a consistent look-and-
Finished GNOME applications work well with the GNOME desktop, but can
be used “standalone” – users only need to install GNOME’s shared libraries.
A
GNOME application isn’t tied to a specific windowing system, GTK+
backends for the X Window System,Wayland, Mac OS X, and even
for a web browser.
At the time of writing, there are new GLib, GTK+ and GNOME stable
every six months, around March and September. A version number has the
form where “minor” is even for stable versions and is
odd for unstable versions. For example, the 3.18.* versions are stable, but the
3.19.* versions are unstable. A new micro stable version (e.g. 3.18.0 → 3.18.1)
doesn’t add new features, only translation updates,bug fixes and performance
improvements. GNOME components are meant to be installed with the same
versions, alongside the version of GTK+ and GLib released at the same time;
it’s for instance a bad idea to run a GNOME daemon in version 3.18 with the
control center in version 3.16. At the time of writing, the latest stable versions
are: GLib GTK+ 3.18 and GNOME al released at the same time
in September 2015. For a a new major version number usually means
there has been an API break, but thankfully previous major versions are parallel-
installable with the new version.During a development cycle (e.g.3.19), there
is no API stability guarantees for new features;but by being an early adopter
your comments are useful to discover more quickly design flaws and bugs.
More information about GNOME:
1. Prerequisite
This book assumes that you already have some programming practice.Here is
a list of recommended prerequisites, with book references.
• This text supposes that you already know the C language. The refer-
ence book is The C Programming Language, by Brian Kernighan and
nis Ritchie
• Object-oriented programming (OOP) is also required for learning GObject.
You should be familiar with concepts like inheritance, an interface, a virtual
method or polymorphism.A good book, with more than sixty guidelines,
is Object-Oriented Design Heuristics, by Arthur Riel
6
• Having read a book on data structures and algorithms is useful, but you
can learn that in parallel.A recommended book is The Algorithm
Manual, by Steven Skiena [4].
• If you want to develop your software on a Unix-like system, another pre-
requisite is to know how Unix works,and be familiar with the command
line, a bit of shell scripting and how to write a Makefile.A possible book
is UNIX for the Impatient, by Paul Abrahams
• Not strictly but highly recommended is to use a version control
system like A good book is Pro Git, by Scott Chacon
The GLib and GTK+ libraries can be used by other programming languages
than C. Thanks to GObject Introspection, automatic bindings are available for a
variety of languages for all libraries based on GObject.
Official GNOME
are available for the following languages: C+ JavaScript Perl, Python and
Val 7 . Vala is a new programming language based on GObject which integrates
the peculiarities of GObject directly in its C#-like Beyond the official
GNOME GLib and GTK+ can be used in more than a dozen
programming languages, with a varying level of support. So why and when
choosing the C language? For writing a daemon on a Unix system,C is the
facto language. But it is less obvious for an application.
To answer the question,
let’s first look at how to structure an application codebase.
A good practice is to separate the graphicaluser interface from the rest of the
application. For a variety of reasons, an application’s graphical interface tends to
be an exceptionally volatile and ever-changing piece of software.It’s the focus of
most user requests for change.It is difficult to plan and execute well the first time
around – often you will discover that some aspect of it is unpleasant to use only
after you have written it. Having several different user interfaces is sometimes
desirable, for example a command-line version, or a web-based interface.
In terms, this means that any large application should have a radical
separation between its various frontends, or interfaces, and the backend. Th
backend should contain all the “hard your algorithms and data struc-
tures, the real work done by the application.Think of it as an abstract
being displayed to and manipulated by the user.
Each frontend should be a “view” and a “controller.” As a “view,” the
must note any changes in the backend,and change the display accordingly.A
a “controller,” the frontend must allow the user to relay requests for change to
the backend (it defines how manipulations of the frontend translate into changes
in the model).
There are many ways to discipline yourself to keep your application separated.
A couple of useful ideas:
7
https://wiki.gnome.org/
Projects/Vala
7
• Write the backend as a library. At the beginning the library can be in-
ternal to the application and statically linked, without API/ABI
guarantees. When the project grows up, and if the code is useful for other
programs, you can turn easily your backend into a shared library.
• Write at least two frontends from the start; one or both can be ugly pro-
totypes, you just want to get an idea how to structure the backend. Re
member, frontends should be easy; the backend has the hard parts.
The C language is a good choice for the backend part of an application. B
using GObject and GObject Introspection, your library will be available for other
projects written in various programming languages.On the other hand, a Python
or JavaScript library cannot be used in other languages.For the frontend(s), a
higher-level language may be more convenient, depending on what languages you
already are proficient with.
If you’re hesitant about the language to choose, here are other aspects to keep in
mind. Note that this text is a little biased since the C language has been chosen.
C is a static-typed language: the variable types and function prototypes in a
program are all known at compilation time.Lots of trivial errors are discovered
by the compiler, such as a typo in a function name. The compiler is also a
great help when doing code refactorings, which is essential for the long-term
maintainability of a program.For example when you split a class in two, if the
code using the initial class is not updated correctly, the compiler will kindly tell
you so8 . With test-driven development and by writing unit tests for
everything, writing a huge codebase in a dynamic-typed language like Python
is also feasible. With a very good code coverage,the unit tests also detect
errors when refactoring the code.But unit tests can be much slower to execute
than compiling the code, since it tests also the program behavior. So it’s
not convenient to run all unit tests when doing code refactorings. Of course
writing unit tests is also a good practice for a C codebase! However for the
GUI part of the writing unit tests is often not a high-priority task if the
application is well tested by its developers.
C is an explicit-typed language:the variable types are visible in the code.It
a way to self-document the code; you usually don’t need to add comments to
explain what the variables contain.Knowing the type of a variable is important
to understand the code, to know what the variable represents and what functions
can be called on it.On a related matter, the object is passed explicitly as a
function argument.Thus when an attribute is accessed through the self pointer,
you know where the attribute comes from.Some object-oriented languages have
the this keyword for that purpose,but it is sometimes optionallike in C++
Java In this latter case, a useful text editor feature is to highlight differently
attributes, so even when the this keyword is not used, you know that it’s
attribute and not a local variable.With the object passed as an argument,
there is no possible confusions.
The C language has a very good toolchain:stable compilers (GCC, Clang, . . .
8
Well kindly is perhaps not the best description, spewing out loads of errors is closer to
reality.
8
text editors (Vim, Emacs, . . . ), debuggers (GDB, Valgrind, . . . ), static-
tools, ..
For some programs, a garbage collector is not appropriate because it pauses the
program regularly to release unused memory.For critical code sections, such as
real-time animations, it is not desirable to pause the program (a garbage col-
lector can sometimes run during several seconds).In this case, manual memory
management like in C is a solution.
Less important, but useful; the verbosity of C in combination with the
GLib/GTK+ conventions has an the code can be searched easily with
a command like grep.For example the functiongtk_widget_show( contains the
namespace (gt ) the class (widget) and the method
) show) With an
oriented language,
k the syntax is generallyobject.show . If “show” is searched
in the code, there will most probably be more
() false positives,so a smarter tool
is needed. Another advantage is that knowing the namespace and the class
of a method can be useful when reading the code, it is another form of self-
documentation.
More importantly, the GLib/GTK+ API documentation is primarily written
the C language. Reading the C documentation while programming in another
language is not convenient.Some tools are currently in development to generate
the API documentation for other target languages, so hopefully in the future it
won’t be a problem anymore.
GLib/GTK+ are themselves written in C. So when programming in C, there is
extra An extra layer is potentially a source of additional bugs and mainte-
nance burdens. Moreover, using the C language is probably better for pedagogi-
cal purposes. A higher-level language can hide some details about
So the code is shorter, but when you have a problem you need to understand not
only how the library feature works, but also how the language binding works.
That said, if (1) you’re not comfortable in C, (2) you’re already proficient with
higher-level language with GObject Introspection support, (3) you plan to write
just a small application or plugin, then choosing the higher-level language makes
perfect sense.
1. Learning
Normally this section should be named “Structure of the Book”, but as you can
see the book is far from finished, so instead the section is named “Learning Path”.
The logical learning path is:
1. The basics of GLib
2. Object-Oriented Programming in C and the basics of GObject;
3. GTK+ and GIO in
Since GTK+ is based on GLib and it is better to understand the
basics of those two libraries first. Some tutorials dive directly into so
after a short period of time you can display a window with some text and three
buttons; it’s but knowing GLib and GObject is not a luxury you want
to understand what you’re doing, and a realistic GTK+ application
9
uses the GLib libraries.GTK+ and GIO can be learned in parallel — once
start using GTK+, you will see that some non-GUI parts are based on
So this book begins with the GLib core library (part I 11) then introduces
Object-Oriented Programming in C (part II p. 35) followed by a Further
chapter (p.
1. The Development
This section describes the development environment typically used when pro-
gramming with GLib and GTK+ on a Unix
On a GNU/Linux distribution, a single package or group can often be installed
to get a full C development environment, including, but not limited to:
• a C89-compatible compiler, GCC for
• the GNU debugger
• GNU
• the Autotools (Autoconf, Automake and
• the man-pages of: the Linux kernel and the glibc9 .
For using GLib and GTK+ as a developer, there are several
• The headers and the documentation can be installed with the package
manager. The name of the packages end typically with one of the following
suffixes: -devel, -dev or - For example glib2-devel and glib2-
on Fedora.
• The latest versions of GLib and GTK+ can be installed with
https://wiki.gnome.org/Projects/
Jhbuild
To read the API documentation of GLib and Devhelp is a handy ap-
plication, if you have installed the -dev or -doc For the text editor
or there are many choices (and a source of many Vim Emacs,
gedit, Anjuta MonoDevelop/Xamarin Studio,Geany, . . A promising special-
ized IDE for GNOME is Builder, currently in For creating a
with you can either write directly the code to do it, or you can use
Glade to design the GUI graphically.Finally, GTK-Doc is used for writing
documentation and add the GObject Introspection annotations.
When using GLib or GTK+, pay attention to not use deprecated APIs for
written code. Be sure that you read the latest documentations. They are also
available online at:
https://
developer.gnome.org/
1.1 Acknowledgment
9
Do not confuse the GNU C Library (glibc) with The former is lower-level.
10
Part
11
Chapter
GLib is the low-level core library that forms the basis for projects such as
and GNOME. It provides data structures, utility functions, portability
and other essential functionality such as an event loop and threads. GLib
available on most Unix-like systems and Windows.
This chapter covers some of the most commonly-used features.GLib is
and the concepts are familiar; so we’ll move quickly.
For more complete coverage
of GLib, see the latest API documentation that comes with the library (for
development environment,see section 1.9 on p. 10) By the if you have
very specific questions about the implementation, don’t be afraid to look at the
source code. Normally the documentation contains enough information,but
you come across a missing detail, please file a bug (of course, the best would be
with a provided patch).
GLib’s various facilities are intended to have a consistent interface;the coding
style is semi-object-oriented, and identifiers are prefixed with “g” to create a
kind of namespace.
GLib has a few toplevel headers:
• glib.h, the main
• gmodule.h for dynamic loading of modules;
• glib-unix.h for Unix-specific
• glib/gi18n.h and glib/gi18n-lib.h for
• glib/gprintf.h and glib/gstdio.h to avoid pulling in all of
Note: instead of reinventing the wheel, this chapter is heavily based on the
corresponding chapter in the book GTK+/Gnome Application Development
Havoc Pennington, licensed under the Open Publication License (see section 1.1
p. 3) GLib has a very stable Despite the fact that Havoc Pennington’s
book was written in 1999 (for GLib 1.2), only a few updates were required to fit
the latest GLib versions (version 2.42 at the time of writing).
12
#include
<glib.h>
MAX (a,
MIN
b); (a,
ABS
b);
CLAMP
(x); (x, low,
high); Listing Familiar C
2. Basic
2.1. Type
Rather than using C’s standard types int , long , etc. GLib defines its own.
These serve a variety of purposes. For example, gint3 is guaranteed to be 32
bits wide, something no standard C89 type can 2ensure. guin is simply easier
to type than unsigned . A few the typedefs exist onlyt for consistency; for
example, gchar is always equivalent to the standardchar .
The most important primitive types defined by GLib:
• gint , guint , gint1 , guint1 , gint3 , guint3 , gint6 , guint6 — these
give you
8 8 integers
6 of a guaranteed
6 2 size.(If
2 it isn’t 4obvious,
4 guin types
are unsigned, the gin types are signed.) t
t
• gboolean is useful to make your code more readable, since C89 has boo
no
type. l
• gchar, gshor , glong, gin , gfloa , gdouble are purely cosmetic.
t t t
• gpointe may be more convenient to type than void * . gconstpointe
gives you const void * . (const gpointer wil not do whatryou typically
r
mean it to; spend some time with a good book on C if you don’t see why.)
• gsiz is an unsigned integer type that can hold the result of the sizeof
operator.
e
13
#include
<glib.h>
GINT_TO_POINTER (p);
GPOINTER_TO_INT (p);
GUINT_TO_POINTER (p);
GPOINTER_TO_UINT (p);
Listing Macros for storing integers in pointers
#include
<glib.h>
g_return_if_fail
g_return_val_if_fail (condition,
(condition);
return_value); Listing Precondition Checks
Most of GLib’s data structures are designed to store a gpointe . If you want
to store pointers to dynamically allocated objects, this ris the right thing.
How-
ever, sometimes you want to store a simple list of integers without having to
dynamically allocate them.Though the C standard does not strictly guarantee
it, it is possible to store gin or guin in gpointe variable on the wide range
of platforms GLib has been t ported
t to; in some
r cases, an intermediate cast is
required. The macros in Listing 2.2 abstract the presence of the cast.
Here’s an example:
gint
gpointer
my_int;
my_pointer;
my_int =
my_pointer = GINT_TO_POINTER
5;
printf ("We are storing %d\n", GPOINTER_TO_INT
(my_int);
(my_pointer));
Be careful, though; these macros allow you to store an integer in a pointer, but
storing a pointer in an integer will not work. To do that you must
store the pointer in along . (It’s undoubtedly a bad idea to do so, however.)
2.1. Debugging
GLib has a nice set macros you can use to enforce invariants and precon-
ditions in your code. GTK+ uses these liberally – one the reasons it’s so
stable and easy to use.They all disappear when you defineG_DISABLE_CHECKS or
G_DISABLE_ASSERT, so there’s no performance penalty in production code. Using
these liberally is a very, very good idea.You’ll find bugs much faster if you do.
You can even add assertions and checks whenever you find a bug to be sure the
bug doesn’t reappear in future versions – this complements a regression suite.
Checks are especially useful when the code you’re writing will be used as a black
box by other programmers; users will immediately know when and how they’ve
misused your code.
Of course you should be very careful to ensure your code isn’t subtly dependent
on debug-only statements to function correctly.Statements that will disappear
in production code should never have side effects.
14
#include
<glib.h>
g_assert
g_assert_not_reached
(condition);
(); Listing Assertions
Without the checks, passing NULL as a parameter to this function would result
in a mysterious segmentation fault.The person using the library would have to
figure out where the error occurred with a debugger, and maybe even dig in to
the GLib code to see what was wrong.With the checks, they’ll get a nice error
message telling them thatNULL arguments are not allowed.
GLib also has more traditional assertion macros, shown in Listing g_assert
2.4.
is basically identical toassert , but responds toG_DISABLE_ASSERT and () behaves
consistently across all ()
platforms. g_assert_not_reached is also provided; this
is an assertion which always fails.()Assertions call abort( to exit the program
and (if your environment supports it) dump a core ) file for debugging purposes.
15
This code from GLib’s calendrical calculations module shows the difference:
GDate *
g_date_new_dmy (GDateDay day
GDateMonth ,month,
GDateYear year
{ )
GDate
g_return_val_if_fail (g_date_valid_dmy (day, month,
*date;
year), NULL);
date = g_new (GDate,
1);
date->julian =
date->dmy =
FALSE;
TRUE;
date->month = month;
date->day =
date->year =
day;
year;
g_assert (g_date_valid
(date));
return
} date;
The precondition check at the beginning ensures the user passes in reasonable
values for the day, month and year;the assertion at the end ensures that GLib
constructed a sane object, given sane values.
g_assert_not_reached should be used to mark “impossible” situations; a com-
mon use is to detect switch statements that don’t handle allpossible values of
()
an enumeration:
switch
(value)
{
case FOO_ONE:
break;
case FOO_TWO:
break;
default:
g_assert_not_reached
} ();
2.1. Memor
16
#include
<glib.h>
gpointer g_malloc (gsize
void g_free (gpointer
n_bytes);
gpointer
mem); g_realloc (gpointer mem, gsize
gpointer
n_bytes);g_memdup (gconstpointer mem, guint
n_bytes); Listing GLib memory allocation
#include
<glib.h>
g_new (type,
g_new0 (type,
count);
g_renew (type, mem,
count);
count); Listing Allocation macros
If it isn’t g_malloc0( fills raw memory with unset bits,not the value
0 for whatever type) you intend to put there. Occasionally someone expects to
get an array of floating point numbers initialized to 0.0;this is guaranteed
to work portably.
Finally, there are type-aware allocation macros, shown in Listing 2.6.
Th typ
argument to each of these is the name of a type, and the count argument ise the
number of typ -size blocks to allocate.These macros save you some typing and
multiplication,
e and are thus less error-prone. They automatically cast to
target pointer type, so attempting to assign the allocated memory to the wrong
kind of pointer should trigger a compiler warning.(If you have warnings turned
on, as a responsible programmer should!)
1
Before the ANSI/ISO C the void * generic pointer type exist, and
malloc( returned a char * value. Nowadays malloc( returns a void * type — which
the same as gpointe — and void * allows implicit
) ) pointer conversions in C. Casting the
of malloc( is needed if: the developer wants to support old compilers; or if the
return value r
developer thinks) that an explicit conversion makes the code clearer; or if a C++ compiler
used, because in C++ a cast from void * type is required.
17
gint g_snprintf (gchar *string, gulong n, gchar const
*format, ...); Listing Portability
#include
<glib.h>
gchar * g_strdup (const gchar
gchar * g_strndup (const gchar *str,
*str);
gchar
gsize * g_strdup_printf (const gchar
n);
gchar * g_strdup_vprintf
*format, ...); (const gchar *format,
gchar
va_list* g_strnfill
args); (gsize length, gchar
fill_char); Listing Allocating Strings
2.1. String
GLib provides a number of functions for string handling; some are unique to
GLib, and some solve portability concerns.They all interoperate nicely with the
GLib memory allocation routines.
For those interested in a better string than
gchar , there’s also a GStrin type.
It isn’t covered in this book, see the API documentation
* for further
g information.
Listing 2.7 shows a substitute GLib provides for snprintf
the function. g_snprintf
wraps native snprintf on platforms that have it,and
() provides an implemen-
()
tation on those that don’t.
()
#include
<glib.h>
gchar * g_strchug (gchar
gchar * g_strchomp (gchar
*string);
gchar * g_strstrip (gchar
*string);
*string); Listing In-place string modifications
18
#include
<glib.h>
gdouble g_strtod (const gchar *nptr, gchar
const gchar * g_strerror (gint
**endptr);
const gchar * g_strsignal (gint
errnum);
signum); Listing 2.10: String Conversions
#include
<glib.h>
gchar * g_strconcat (const gchar
gchar * g_strjoin
*string1, ...); (const gchar
*separator, ...); Listing 2.11: Concatenating Strings
Listing 2.10 shows a few more semi-standard functions GLib wraps. g_strto is
like strtod – it converts string npt to a double – with the exception d that
wil ()
also attempt to convert the double
r in the"C locale if it fails to convert
in the user’s default locale.*endptr is set to the
" first unconverted character, i.e.
any text after the number representation. If conversion fails, *endptr is set to
npt . endptr may be NULL, causing it to be ignored.
r
g_strerror and g_strsigna are like their non-g_ equivalents, but portable.
(They return a string
() l() representation for anerrn or a signal number.)
o
GLib provides some convenient functions for concatenating strings, shown in
Listing 2.11. g_strconcat returns a newly-allocated string created by concate-
nating each of()the strings in the argument list.
The last argument must beNULL,
so g_strconcat knows when to stop. g_strjoi is similar, separato is
inserted between each string.If
() n()
separato is NULL, no separator is used.
r
r
Finally, Listing 2.12 summarizes a few routines which manipulateNULL-terminated
arrays of strings. g_strspli breaks strin at each delimite , returning a
newly-allocated array.
t() g_strjoin concatenates
g each string
r in the array with
an optional separato , returning an allocated string. g_strfreev frees each
v()
string in therarray and then the array itself. ()
#include
<glib.h>
gchar ** g_strsplit (const gchar
*string, const gchar
gint
*delimiter,
gchar * g_strjoinv (const gchar *separator, gchar
max_tokens);
void g_strfreev (gchar
**str_array);
**str_array);
Listing 2.12: Manipulating NULL-terminated string vectors
19
typedef struct _GSList
GSList;
struct
_GSList
{
gpointer
GSList
data;
} *next;
; Listing 2.13: GSLis cell
t
#include
<glib.h>
GSList * g_slist_append (GSList *list,
GSList
gpointer * g_slist_prepend
data); (GSList *list,
GSList
gpointer * g_slist_insert
data); (GSList *list, gpointer data,
GSList * g_slist_remove (GSList *list,
gint position);
gconstpointer data);
Listing 2.14: Changing linked list contents
2. Data
GLib implements many common data structures, so you don’t have to reinvent
the wheel every time you want a linked list. This section covers GLib’s
mentation of linked lists, sorted binary trees, N-ary trees, and hash tables.
2.2. List
GLib provides generic single and doubly linked lists,GSLis and GLis , respec-
tively These are implemented as lists ofgpointe ; yout can use them
t to hold in-
tegers with theGINT_TO_POINTER and r
GPOINTER_TO_INT macros. GSLis and GLis
have almost the same API’s, except that there is ag_list_previou
t function
t
and no g_slist_previou . This section will discuss GSLis but everything also
s()
applies s()
to the doubly linked list. t
20
#include
<glib.h>
typedef void (* GDestroyNotify) (gpointer
data);
void g_slist_free (GSList
void g_slist_free_full (GSList *list, GDestroyNotify
*list);
free_func); Listing 2.15: Freeing entire linked lists
so the append, insert, and remove functions run in O(n) time, with n the length
of the list.
GLib will handle memory issues, deallocating and allocating list cells as needed.
For example, the following code would remove the above-added element and
empty the list:
list = g_slist_remove (list,
element);
li is now NULL. You still have to freeelement yourself, of course.
st
To access a list element, you refer to theGSLis struct
gchar *my_data = list- t
>data;
To iterate over the list, you might write code like this:
GSList
*l;
for (l = list; l != NULL; l
{
= l->next)
gchar *str = l-
g_print ("Element: %s\n",
>data;
} str);
21
#include
<glib.h>
typedef void (* GFunc) (gpointer data, gpointer
user_data);
GSList * g_slist_find (GSList *list,
GSList * g_slist_nth
gconstpointer data); (GSList *list,
gpointer
guint n); g_slist_nth_data (GSList *list,
GSList
guint n);* g_slist_last
gint
(GSListg_slist_index
*list); (GSList *list,
void g_slist_foreach
gconstpointer data); (GSList *list, GFunc func, gpointer
user_data); Listing 2.16: Accessing data in a linked list
To use this function, you would store the list and its end somewhere,and pass
their address to efficient_appen :
GSList* list = d()
GSList* list_end =
NULL;
NULL;
efficient_append (&list, &list_end, g_strdup
efficient_append (&list, &list_end, g_strdup
("Foo"));
efficient_append (&list, &list_end, g_strdup
("Bar"));
("Baz"));
Of course you have to be careful not to use any list functions that might change
the end of the list without updatinglist_e .
nd
For accessing list elements, the functions in Listing 2.16 are provided. None
of these change the list’s structure. g_slist_foreac applies a GFunc to each
element of the list. h()
2
A more convenient way is to use the GQueue data type: a double-ended queue that keeps
a pointer to the head, a pointer to the tail, and the length of the doubly linked list.
22
Used in g_slist_foreac , your GFunc will be called on eachlist- in li ,
passing h()
the user_dat you provided tog_slist_foreac . g_slist_foreac
>data st is
comparableato Scheme’s “map” function.
h() h()
For example, you might have a list of strings, and you might want to be able
to create a parallel list with some transformation applied to the strings.
Here is
some code, using theefficient_appen function from an earlier example:
d()
typedef struct _AppendContext AppendContext;
struct _AppendContext
{
GSList
GSList
*list;
const gchar *append;
*list_end;
}
;
static void
append_foreach (gpointer
data, gpointer
{ user_data)
gchar *oldstring =
AppendContext
data; *context =
user_data;
efficient_append (&context-
>list, &context-
g_strconcat
>list_end, (oldstring, context->append,
} NULL));
GSList
c
*opy_with_append *list_of_stri
(GSList const gchar *append)
ngs,
{
AppendContext
context;
context.list =
context.list_end
NULL; =
context.append =
NULL;
append;
g_slist_foreach (list_of_strings, append_foreach,
&context);
return
} context.list;
GLib and GTK+ use the “function pointer and user data” idiom If you
have functional programming experience, this is much like using lambda expres-
sions to create a closure.(A closure combines a function with an environment –
a set of name-value bindings.In this case the “environment” is the user data you
pass to append_foreach( , and the “closure” is the combination of the function
pointer )and the user data.)
There are some handy list-manipulation routines,listed in Listing Wit
the exception of g_slist_cop , al of these affect the lists in-place. Whic
means you must assign
y() the return value and forget about the passed-in pointer,
just as you do when adding or removing list elements. g_slist_cop returns
a newly-allocated list, so you can continue to use both lists and must free both
y()
lists eventually.
Finally, there are some provisions for sorted lists, shown in Listing 2.18.
To use
23
#include
<glib.h>
guint g_slist_length (GSList
GSList
*list);* g_slist_concat (GSList *list1,
GSList * g_slist_reverse (GSList
*list2);
GSList
*list); g_slist_copy (GSList
*
*list); Listing 2.17: Manipulating a linked list
#include
<glib.h>
typedef gint (* GCompareFunc) (gconstpointer a,
gconstpointer b);
GSList * g_slist_insert_sorted (GSList *list, gpointer data,
GSList * g_slist_sort
GCompareFunc func); (GSList *list, GCompareFunc
GSList * g_slist_find_custom (GSList *list, gconstpointer data,
compare_func);
GCompareFunc func); Listing 2.18: Sorted lists
these, you must write aGCompareFunc, which is just like the comparison function
in the standard qsort .
()
If a < , the GCompareFunc should return a negative value; if a > a positive
value;
b if a == it should return 0. b
b
Once you have a comparison function, you can insert an element into an already-
sorted list, or sort an entire list.
Lists are sorted in ascending order.
You can even
recycle your GCompareFunc to find list elements, usingg_slist_find_custo .
m()
Be careful with sorted lists;misusing them can rapidly become very inefficient.
For example, g_slist_insert_sort is an O(n) operation, but you use it
ed() multiple elements the loop runs in quadratic time (O(n2 ))
in a loop to insert
It’s better to simply prepend all your elements, and then call g_slist_sor .
g_slist_sor runs in O(n log n). t()
t()
You can also use theGSequence data structure for sorted data.GSequence has an
API of a list, but is implemented internally with a balanced binary tree.
2.2. Tree
There are two different kinds of tree in GLib;GTree is your basic balanced binary
tree, useful to store key-value pairs sorted by key; GNode stores arbitrary tree-
structured data, such as a parse tree or taxonomy.
GTre
To create and destroy aGTree, use a constructor and destructor displayed in List-
ing 2.19. GCompareFunc is the same qsort -style comparison function described
for GSLis ; in this case it’s used to()
compare keys in the tree.
g_tree_new_ful
is useful to ease memory management for dynamically-allocated
t l() keys and values.
Th GTree struct is an opaque data type. Its content is accessed and modified
only with public functions.
24
#include
<glib.h>
typedef gint (* GCompareFunc) (gconstpointer a,
typedef gint (*b);
gconstpointer GCompareDataFunc) (gconstpointer
a, gconstpointer
gpointer
b,
user_data);
GTree * g_tree_new (GCompareFunc
key_compare_func);
GTree * g_tree_new_full (GCompareDataFunc
key_compare_func, gpointer
GDestroyNotify
key_compare_data,
GDestroyNotify
key_destroy_func,
value_destroy_func);
void g_tree_destroy (GTree
*tree); Listing 2.19: Creating and destroying balanced binary trees
#include
<glib.h>
void g_tree_insert (GTree *tree, gpointer key, gpointer
gboolean g_tree_remove (GTree *tree, gconstpointer
value);
gpointer g_tree_lookup (GTree *tree, gconstpointer
key);
key); Listing 2.20: Manipulating GTree contents
Functions for manipulating the contents ofthe tree are shown in Listing 2.20.
Al very straightforward; g_tree_inser overwrites any existing value, so if
you don’t use g_tree_new_ful
t() , be careful if the existing value is your only
pointer to a chunk
l() of allocated memory. If g_tree_lookup fails to find the
key, it returns NULL, otherwise it returns the ()
associated value. Both keys and
values have type gpointe or gconstpointe , but the GPOINTER_TO_INT() and
GPOINTER_TO_UINT() r macros allow
r you to use integers instead.
There are two functions which give you an idea how large the tree is,shown in
Listing 2.21.
Using g_tree_foreach (Listing 2.22) you can walk the entire tree. To use
you provide
() a GTraverseFunc, which is passed each key-value pair and a data
argument you give to g_tree_foreach . Traversal continues as long as the
GTraverseFunc returns FALSE
() ; if it ever returns TRUE then traversal stops. Yo
can use this to search the tree by value.
GNod
#include
<glib.h>
gint g_tree_nnodes (GTree
gint g_tree_height (GTree
*tree);
*tree); Listing 2.21: Determining the size of aGTree
25
#include
<glib.h>
typedef gboolean (* GTraverseFunc) (gpointer
key, gpointer
gpointer
value,
data);
void g_tree_foreach (GTree *tree, GTraverseFunc func, gpointer
user_data); Listing 2.22: Traversing a GTree
struct _GNode
{
gpointer
GNode *next;
data;
GNode *prev;
GNode *parent;
GNode
} *children;
; Listing 2.23: GNode cell
#include
<glib.h>
g_node_prev_sibling
g_node_next_sibling
(node);
g_node_first_child
(node);
(node); Listing 2.24: Accessing GNode
26
#include
<glib.h>
GNode * g_node_new (gpointer
data); Listing 2.25: Creating a GNode
#include
<glib.h>
GNode * g_node_insert (GNode *parent, gint position, GNode
GNode
*node);* g_node_insert_before (GNode *parent, GNode *sibling, GNode
GNode
*node);* g_node_prepend (GNode *parent, GNode *node);
Listing 2.26: Building a GNode tree
#include
<glib.h>
g_node_append (parent,
g_node_insert_data (parent, position,
node);
g_node_insert_data_before
data); (parent, sibling,
g_node_prepend_data
data); (parent,
g_node_append_data
data); (parent,
data); Listing 2.27: Building a GNode
#include
<glib.h>
void g_node_destroy (GNode
void g_node_unlink (GNode
*root);
*node); Listing 2.28: Destroying a GNode
27
#include
<glib.h>
G_NODE_IS_ROOT (node);
G_NODE_IS_LEAF
(node); Listing 2.29: Predicates for GNode
#include
<glib.h>
guint g_node_n_nodes (GNode *root, GTraverseFlags
GNode * g_node_get_root (GNode
flags);
gboolean g_node_is_ancestor (GNode *node, GNode
*node);
guint g_node_depth (GNode
*descendant);
GNode * g_node_find (GNode
*node);
*root, GTraverseType
GTraverseFlags
order,
gpointer
flags,
data);
Listing 2.30: GNode properties
g_node_unlink( removes a node and makes it into a root node; i.e., it converts
a subtree into an independent tree.
)
There are two macros for detecting the top and bottom of aGNode tree, shown
in Listing A root node is defined as a node with no parent or siblings.A
leaf node has no children.
You can ask GLib to report information about a GNode, including the
number of nodes it contains, its root node, its depth, and the node containing a
particular data pointer.These functions are shown in Listing 2.30.
GTraverseType is an enumeration; there are four possible values. Here are their
meanings:
• G_PRE_ORDER visits the current node, then recurses each child in turn.
• G_POST_ORDER recurses each child in order, then visits the current node.
• G_IN_ORDER first recurses the leftmost child of the node, then visits the node
itself, then recurses the rest of the node’s children.This isn’t very
mostly it is intended for use with a binary tree.
• G_LEVEL_ORDER first visits the node itself; then each of the node’s children;
then the children of the children;then the children of the children of the
children; and so on.That is, it visits each node of depth 0, then each node
of depth 1, then each node of depth 2, etc.
GNode’s tree-traversal functions have aGTraverseFlag argument. This is a
field used to change the nature of the straversal.Currently there are only three
flags – you can visit only leaf nodes, only non-leaf nodes, or all nodes:
• G_TRAVERSE_LEAVES means to traverse only leaf nodes.
• G_TRAVERSE_NON_LEAVES means to traverse only non-leaf nodes.
• G_TRAVERSE_ALL is simply a shortcut for
(G_TRAVERSE_LEAVES | G_TRAVERSE_NON_LEAVES).
28
#include
<glib.h>
typedef gboolean (* GNodeTraverseFunc) (GNode *node, gpointer
data); void (* GNodeForeachFunc) (GNode *node, gpointer
typedef
data);
void g_node_traverse (GNode
*root, GTraverseType
GTraverseFlags
order,
gint
flags,
GNodeTraverseFunc
max_depth,
gpointer
func,
data);
void g_node_children_foreach (GNode
*node, GTraverseFlags
GNodeForeachFunc func,
flags,
gpointer
data);
guint g_node_max_height (GNode
void g_node_reverse_children (GNode
*root);
guint g_node_n_children (GNode
*node);
gint g_node_child_position (GNode *node, GNode
*node);
GNode * g_node_nth_child (GNode *node, guint
*child);
GNode * g_node_last_child (GNode
n);
*node); Listing 2.31: Accessing a GNode
Listing 2.31 shows some of the remainingGNode functions. They are straightfor-
ward; most of them are simply operations on the node’s list of children.
There are
two function typedefs unique to
GNode : GNodeTraverseFunc and GNodeForeachFunc .
These are called with a pointer to the node being visited, and the user data you
provide. A GNodeTraverseFunc can return TRUE to stop whatever traversalis
progress; thus you can useg_node_traverse to search the tree by value.
()
2.2. Hash
29
#include
<glib.h>
typedef guint (* GHashFunc) (gconstpointer
key); gboolean (* GEqualFunc) (gconstpointer a,
typedef
typedef void (* GDestroyNotify)
gconstpointer b); (gpointer
data);
GHashTable * g_hash_table_new (GHashFunc hash_func, GEqualFunc
key_equal_func);
GHashTable * g_hash_table_new_full (GHashFunc
hash_func, GEqualFunc
GDestroyNotify
key_equal_func,
GDestroyNotify
key_destroy_func,
value_destroy_func);
void g_hash_table_destroy (GHashTable
*hash_table);Listing 2.32: GHashTable constructors and destructor
#include
<glib.h>
guint g_int_hash (gconstpointer
gboolean g_int_equal (gconstpointer key1, gconstpointer
key);
guint g_direct_hash (gconstpointer
key2);
gboolean g_direct_equal (gconstpointer key1, gconstpointer
key);
guint
key2);g_str_hash (gconstpointer
gboolean
key); g_str_equal (gconstpointer key1, gconstpointer
key2); Listing 2.33: Pre-written hashes/comparisons
Manipulating the hash table is simple. The routines are summarized in List-
ing 2.34. Insertions do not copy the key or value; these are entered into the table
exactly as you provide them, replacing any pre-existing key-value pair with the
same key (“same” is defined by your hash and equality functions, remember). If
this is a problem, you must do a lookup or remove before you insert.Be espe-
cially careful if you dynamically allocate keys or values. If you have provided
GDestroyNotif functions, those will be called automatically on the old key-value
pair before replacing it.
y
30
#include
<glib.h>
gboolean g_hash_table_insert (GHashTable *hash_table, gpointer key, gpointer
gboolean
value); g_hash_table_remove (GHashTable *hash_table, gconstpointer
gpointer
key); g_hash_table_lookup (GHashTable *hash_table, gconstpointer
gboolean
key); g_hash_table_lookup_extended (GHashTable
*hash_table, gconstpointer
gpointer
lookup_key,
gpointer
*orig_key,
*value);
Listing 2.34: Manipulating a GHashTable
31
each source of events. When an event arrives, the event loop dispatches it to
the application. The event can then be taken into account,either in the same
thread or another thread. The Figure 2.1 shows a high-levelview of what is
main event loop.
Th main( function of an event-driven application looks like:
gin )
main
t argc
(gint gchar ,
{ *argv[])
/* Create main window and attach signal
callbacks. */
/* Run the main event
gtk_main
loop. */
();
return
} 0;
The GTK+ event loop is slightly higher than the GLib event loop
straction. gtk_main( runs the main event loop untilgtk_main_quit is called.
)
gtk_main_quit is typically called in the function callback
() when the close button
is clicked or the Quit menu action is activated.
()
static void
create_button (MyObject
{
*self)
GtkButton
*button;
/* Create
button */
/* Attach signal
g_signal_connect
callback */
(button, "clicked
G_CALLBACK
",
self
(button_clicked_cb),
} );
32
#include
<glib.h>
typedef gboolean (* GSourceFunc) (gpointer
user_data);
guint g_idle_add (GSourceFunc function, gpointer
guint
data);g_timeout_add (guint interval, GSourceFunc function,
gpointer data);
gboolean g_source_remove (guint
source_id); Listing 2.35: Idles and timeouts
For the second solution, GLib provides the g_idle_add and g_timeout_add(
functions (see Listing 2.35).An idle function()will be called when
) the main loop
is idle, that when the main loop has nothing else to do.A timeout function
is called at regular intervals.The boolean return value of aGSourceFunc permits
to continue or stop the function.If it continues, the function will be called again
by the main loop, at the next idle time or timeout. You can manually remove
the GSourceFunc by calling g_source_remove( , which takes as the parameter the
source ID as returned byg_idle_add
) or g_timeout_add( . You must pay atten-
tion to remove aGSourceFunc
() when the object
) on which you do the computation
is destroyed. So you can store the source ID in an object attribute, and call
g_source_remove( in the destructor if the source ID is different from 0. (See the
GObject library to create your own classes in
)
2. Other
There simply isn’t space to cover all of GLib’s features in this It’s
looking at GLib whenever you find yourself thinking, “There really should
a function that...”. This section lists other features GLib provides, but is
exhaustive.
Some core application support not already mentioned:
• GErro – an error reporting system, similar to exceptions in other languages.
r
• g_log facility allows you to print warnings, messages, etc. with
The
configurable
() log levels and pluggable print routines.
Utilities
• A commandline option parser.
• A unit-test framework.
• A timer
• Calendrical/date-arithmetic functions.
• Filename manipulation, such asg_path_get_basename() and g_path_is_absolute .
()
• A simple XML
• Perl-compatible regular expressions.
A selection of smaller utilities:
• G_MAXFLOAT, equivalents for many numeric types.
33
• Byte-order conversions.
• G_DIR_SEPARATOR handles Windows/Unix differences.
34
Part
Object-Oriented
in
35
Introduction to Part
Now that you’re familiar with the GLib core library, what is the next step?
the Learning Path section explained (section 1.8 p.9) the logical follow-up is
Object-Oriented Programming (OOP) in C and the basics of
Every GTK+ widget is a subclass the GObject base class. So knowing the
basic concepts of GObject is important for using a GTK+ widget or
GObject-based utility, but also for creating your own GObject classes.
It is important to note that although the C language is not object-oriented,i
is possible to write “semi-object-oriented” C code easily, without GObject.For
learning purposes,that’s what this part begins GObject is then easier to
learn. What GObject adds is more features such as reference counting,inheri-
tance, virtual functions, interfaces, signals and more.
But why following an object-oriented style in the first place? An object-oriented
code permits to avoid global variables. And if you have read any sort of program-
ming best-practices guide, you know that you should, if possible, avoid global
variables1 . Because using global data makes the code harder to manage and
understand, especially when a program becomes larger.It also makes the code
more difficult to re-use. It is instead better to break a program into smaller,
self-contained pieces, so that you can focus on only one part of the code at a
time.
This part of the book comprises two chapters:
• Chapter 3, which explains how to write your own semi-OOP classes;
• Chapter 4, which explains the basics of GObject.
1
A global variable in C can be static variable declared at the top of a *.c file, that can
thus be accessed from any function in that *.c file. This is sometimes useful, but should be
avoided if possible. There is another kind of global variable in C: an extern variable that can
be accessed from any *.c file. The latter is much worse than the former.
36
Chapter
Semi-Object-
Programming in
It was explained in the previous chapter that the GLib core library uses a semi-
object-oriented coding style. This section explains what it means, and how to
write your own code with this coding style.
One of the main ideas of OOP is to keep related data and behavior in one 1 .
In the data is stored in a struct , and the behavior is implemented with
functions. To keep those in one place, we put them in the same *.c file, with the
public functions present in the corresponding *.h file (the header).
Another important idea of OOP is to hide all data within its In C, it
that the struct full declaration should be present only in the *.c file, while the
header contains only a typedef . How the data is stored within the class and
which data structures are used should remain an implementation detail. A user
of the class should not be aware of the implementation details, it should instead
rely only on the external interface, that is, what is present in the header and the
public documentation. That the implementation of the class can change
without affecting the users of the class, as long as the API doesn’t change.
3. Header
3.1. Project
The first thing to note is the use of the namespace “Myapp”. Each symbol in
the header is prefixed by the project namespace.
It is a good practice to choose a namespace for your code, to avoid symbol
conflicts at link It is especially important to have a namespace for a library,
but it is also better to have one for an application. Of course the namespace
1
This is one of the guidelines discussed in Object-Oriented Design Heuristics [2].
2 https://wiki.gnome.org/Projects/
gspell
37
#ifndef MYAPP_SPELL_CHECKER_H
#define MYAPP_SPELL_CHECKER_H
#include
<glib.h>
G_BEGIN_DECLS
MyappSpellChecker *
myapp_spell_checker_new (const gchar
*language_code);
void
myapp_spell_checker_fre (MyappSpellChecker
e *checker);
gboolean
myapp_spell_checker_check_word (MyappSpellChecker
const gchar
*checker, *word,
gssiz word_length)
e ;
GSList
* myapp_spell_checker_get_suggestions (MyappSpellChecker
*checker, const gchar *word,
gssiz word_length)
e ;
G_END_DECLS
#endif /* MYAPP_SPELL_CHECKER_H
*/ Listing myapp-spell-checker.h
38
needs to be unique for each codebase; for instance you must not re-use the “G”
or “Gtk” namespaces for your application or library!
3.1. Class
Additionally, there is a second namespace with the class name, here “SpellChecker”.
If you follow that convention consistently, the name of a symbol is more pre-
dictable, which makes the API easier to work with.The name of a symbol will
always be “project namespace” + “class name” + “symbol
3.1. Include
/
* ...
#endif /* MYAPP_SPELL_CHECKER_H
*/
It protects the header from being included several times in the same *.c file.
3.1. C++
3.1. #includ
39
myapp-spell-checker.h contains the #include :
#include
<glib.h>
Because glib.h is needed for G_BEGIN_DECLS and G_END_DECLS macros, for
the GLib basic type definitions gchar, gboolean, etc) GSLis .
t
If the #include in myapp-spell-checker.h is each *.c file that in-
cludes myapp-spell-checker.h would also need to include glib.h
otherwise the compiler would not be able to compile that *.c file.
But we
want to add such requirement for the users of the class.
3.1. Type
3.1. Object
3.1. Object
40
3.1.1 Other Public
#include "myapp-spell-
#include
checker.h"
<string.h>
struct _MyappSpellChecker
{
gchar
*language_code;
/* Put here other data structures used to
* the spell
implement
*
checking.
} /
;
static void
load_dictionary (MyappSpellChecker
{
*checker)
/
} * ...
;
/
***
* @language_code: the language code to
myapp_spell_checker_new
*
use.
* Returns: a new #MyappSpellChecker object.
*
Free with
*
myapp_spell_checker_free
MyappSpellChecker
/ *
myapp_spell_checker_new (const gchar
{
*language_code)
MyappSpellChecker
*checker;
g_return_val_if_fail (language_code != NULL,
NULL);
checker = g_new0 (MyappSpellChecker,
checker->language_code = g_strdup
1);
(language_code);
load_dictionary
(checker);
41
return
} checker;
/
***
* @checker: a
myapp_spell_checker_fre
*
#MyappSpellChecker.
* Frees
*
@checker.
void
/
myapp_spell_checker_free (MyappSpellChecker
{
*checker)
if (checker ==
return;
NULL)
g_free (checker-
g_free
>language_code);
} (checker);
/
***
* @checker: a
myapp_spell_checker_check_word
* @word: the word to
#MyappSpellChecker.
* @word_length: the byte length of @word, or -1 if @word is
check.
*
nul-terminated.
* Returns: %TRUE if @word is correctly spelled, %FALSE
*
otherwise.
gboolean
/
myapp_spell_checker_check_word (MyappSpellChecker
*checker, const gchar *word,
gssiz word_length
{ e )
g_return_val_if_fail (checker != NULL,
g_return_val_if_fail (word != NULL,
FALSE);
g_return_val_if_fail (word_length >= -1,
FALSE);
FALSE);
/* ... Check if the word is present in a
dictionary. */
return TRUE;
}
/
***
* @checker: a
myapp_spell_checker_get_suggestio
* @word: a misspelled
#MyappSpellChecker.
* @word_length: the byte length of @word, or -1 if @word is
word.
*
nul-terminated.
* Gets the suggestions for @word. Free the return
*
value with
*
g_slist_free_full(suggestions,
* Returns: (transfer full) (element-type utf8): the
*
list of suggestions.
GSList
/
myapp_spell_checker_get_suggestions (MyappSpellChecker
*
*checker, const gchar *word,
gssiz word_length
{ e )
42
GSList *suggestions =
NULL;
g_return_val_if_fail (checker != NULL,
g_return_val_if_fail
NULL); (word != NULL,
g_return_val_if_fail
NULL); (word_length >= -1,
NULL);
if (word_length ==
-1)word_length = strlen
(word);
if (strncmp (word, "punchness", word_length)
==suggestions
0) = g_slist_prepend
(suggestions, g_strdup
("punchiness"));
return
} suggestions;
Listing myapp-spell-checker.c
3.2. Order of
At the top of the file, there is the usual list#include ’s A small but noteworthy
detail is that the include order was not chosen at random. In a certain
file, it is better to include first its corresponding *.h file, and then the other
headers5 . By including first myapp-spell-checker.h, if #include is missing
in myapp-spell-checker.h, the compiler will report an As explained in
section 3.1.6 p. 39, a header should always have the minimum required #include ’
for that header to be included in turn.
Also since glib.h is already included in myapp-spell- there is no
need to include it a second time in myapp-spell-
3.2. GTK-Doc
43
general structure with the list of symbols and optionally provide additional con-
tent written in the DocBook XML Those files are usually present in the
docs/reference/
directory.
Describing in detail how to integrate GTK-Doc support in your code is beyond
the scope of this book.For that, you should refer to the GTK-Doc manual
Every GLib-based library should have a GTK-Doc But it is
useful to write a GTK-Doc documentation for the internal code of an application.
As explained in section 1.7.1 p. 7, it is a good practice to separate the backend of
an application from its frontend(s), and write the backend as an internal library,
or later a shared library. As such, it is recommended to document the public
API of the backend with GTK-Doc, even if it is still an internal, statically-
library. Because when the codebase becomes larger, it is a great help – especially
for newcomers – to have an overview of the available classes,and to know how
to use a class without the need to understand its implementation.
In the example code, you can see that theload_dictionar function has been
marked as static . It is in fact a good practice
y() in C to mark internal
as static . A static function can be used only in the same *.c file.
On the other
hand, a public function should be non-static and have a prototype in a header.
44
There is the -Wmissing-prototypes GCC warning option to ensure that a
of code follows this convention6 .
Also contrarily to a public function, a static function doesn’t require to be
prefixed by the project and class namespaces (here,myapp_spell_checke )
r
3.2. Defensive
Note that theg_retur macros should be used only for the entry points of a class,
that in its npublic functions. You can usually assume that the parameters
passed to a static function are valid, especially the selfparameter. However,
is sometimes useful to check an argument of astatic function with g_assert ,
to make the code more robust and self-documented. ()
3.2. Coding
Finally it is worth explaining a few things about the coding style. You are
encouraged to use early on the same coding style for your project, because it can
be something difficult to change afterwards.If every program has different code
conventions, it’s a nightmare for someone willing to contribute.
Here is an example of a function definition:
gboolean
myapp_spell_checker_check_word (MyappSpellChecker
*checker, const gchar *word,
gssiz word_length
{ e )
/
} * ...
See how the parameters are aligned: there is one parameter per line, with the
type aligned on the opening parenthesis, and with the names aligned on the same
column. Some text editors can be configured to do that automatically.
For a function call, if putting all the parameters on the same line results in a too
long line, the parameters should also be aligned on the parenthesis, to make the
code easier to read:
function_call
(one_param, another_param,
yet_another_param)
;
Unlike other projects like the Linux kernel, there is not really a limit on
line length. A GObject-based code can have quite lengthy lines, even if the
parameters of a function are aligned on the parenthesis.Of course if a line ends
6
When using the Autotools, the AX_COMPILER_FLAGS Autoconf macro enables, among other
things, that GCC
45
at say, column 120, it might mean that there are too many indentation levels
and that you should extract the code into intermediate functions.
The return type of a function definition is on the previous line than the function
name, so that the function name is at the beginning of the line.When writing
a function prototype, the function name should never be at the beginning of a
line, it should ideally be on the same line as the return type7 . That you
can do a regular expression search to find the implementation of a function, for
example with the grep shell command:
$ grep -n -E "^function_name"
*.c
Or if the code is inside a Git repository, it’s a little more convenient to use
git :
grep
$ git grep -n -E
"^function_name"
Likewise, there is an easy way to find the declaration of a public struct . Th
convention is to prefix the type name by an underscore when declaringstructthe .
For example:
/* In the
typedef */ _MyappSpellChecker MyappSpellChecker;
header:struct
/* In the *.c
file: _MyappSpellChecker
struct */
{
/
} * ...
;
As a to find the full declaration of theMyappSpellChecker type, you can
search “_MyappSpellChecker ”
$ git grep -n
_MyappSpellChecker
In GLib/GTK+, this underscore-prefix convention is normally applied to
struct that has a separate typedef line. The convention is not thoroughly fol-
lowed when a struct is used in only one *.c file. And the convention is usually
not followed for enum types. However, for your project, nothing prevents you
from applying consistently the underscore-prefix convention to all types.
Note that there exists more sophisticated tools than grep to browse a C codebase,
for example Cscope8 .
To learn more about the coding style used in GLib, GTK+ and GNOME,
the GNOME Programming Guidelines [11].
7
In myapp-spell- the function names are indented with two spaces instead of
being on the same lines as the return types, because there is not enough horizontal space on
the page.
8
http://cscope.sourceforge.net/
46
Chapter
A Gentle Introduction
GObjec
4. Inheritanc
47
The GObject library provides GObject base class. Every class in GIO
GTK+ inherit — directly or indirectly — from GObject base class. When
looking at a GObject-based class, the documentation (if written with GTK-
always contains an Object Hierarchy.For instance, the GtkApplicatio has the
following object hierarchy: n
GObject
GApplicatio
n GtkApplicatio
n
It means that when you create a GtkApplicatio object, you also have access
to the functions, signals and properties
n of GApplicatio (implemented in
and GObject. Of course, the g_application n functions take as first argument a
variable of type GApplication_* ”, not GtkApplication ”. To cast the variable
to the good type, the recommended way
* * is to use the G_APPLICATION() macro.
For example:
GtkApplication
*app;
g_application_mark_busy (G_APPLICATION
(app));
4. GObject
48
4. Interfaces
The two macros explained in the previous section work for interfaces too. A
example with GtkGri :
GtkWidget d
*vgrid;
vgrid = gtk_grid_new
gtk_orientable_set_orientation
(); (GTK_ORIENTABLE
(vgrid), GTK_ORIENTATION_VERTICAL);
So when you search a certain feature in the API for a certain GObject class, the
feature can be located at three different places:
• In the GObject class
• In one of the parent classes in the Object Hierarchy;
• Or in one of the Implemented
4. Reference
49
Figure 4.1: Using a weak reference to break the reference cycle between A and
B
needed, usually in the class destructor.A weak reference must be removed with
g_object_remove_weak_pointer( or g_object_weak_unref . So in Figure
the destructor of class B must remove
) () the weak reference if it is not
done.
4.4. Floating
The Listing 4.1 p. 52 shows how memory management is handled with a normal
GObject Compare this to the Listing 4.2, which shows how memory manage-
ment is handled with a GObject deriving fromGInitiallyUnowne . The differ-
ence is that g_object_unref is not called in the
d latter so it shortens
the code. ()
GObject
GInitiallyUnowne
d GtkWidget
GtkEntr
y
4. Signals and
A GObject class can emit With the GLib main event loop
explained in section 2.3 p. 31), this is the foundation for event-driven program-
ming. An example of a signal is when the user clicks on a button.
The application
connects a callback function to the signal to perform the desired action when
the event occurs.
Another concept of GObject are properties, which is related to signals.
A
erty is basically an instance variable surmounted with a
"notif signal that is
emitted when its value changes. A good example of a property
y" is the state of
a check button, i.e. a boolean value describing whether the button is currently
checked or not. When the state changes, the"notif signal is sent.
y"
50
To create your own signals or properties, a GObject sub-class must be created.
As explained in the introduction of this chapter, this is beyond the scope of
this book, but you should be aware that creating your own signals or properties
is of course possible, and recommended. In creating a GObject signal or
property is a nice way to implement the Observer design pattern [3]; that
one or several objects observing state changes of another object,by connecting
function callbacks. The object emitting the signal is not aware of which objects
receive the signal. GObject just keeps track of the list of callbacks to So
adding a signal permits to decouple classes.
51
/* Normal GObject
*/
a_normal_gobject = normal_gobject_new
/* a_normal_gobject has now a reference count
();
of 1. */
container_add (container,
/* a_normal_gobject has now a reference count
a_normal_gobject);
of 2. */
/* We no longer need a_normal_gobject, so we
g_object_unref
unref it. */
/* a_normal_gobject has now a reference count
(a_normal_gobject);
of 1. */ Listing Memory management of normal GObjects.
void
user_function (GtkButton
*button, gpointe user_data
Listingr );
The prototype of theGtkButton::click signal.
ed
52
static void
button_clicked_cb (GtkButton
*button, gpointe user_data
{ r )
MyClass *my_class = MY_CLASS
(user_data);
g_message ("Button
} clicked!");
static void
create_button (MyClass
{
*my_class)
GtkButton
*button;
/* Create the
/
button */
* ...
/* Connect the callback
g_signal_connect
function */
(button, "clicked
G_CALLBACK
",
my_class
(button_clicked_cb),
} );
Listing How to connect to a signal
53
The Listing 4.5 shows a complete example of how to disconnect a signal handler
when its user_dat argument is freed. We come back to an example with a
spell checker,
a because the example with the GTK+ button doesn’t fit the
3
situation .
Th user_dat callback argument is a MyTextView instance, with MyTextView
implemented
a with a semi-OOP style. Since the spell checker object can live
longer than the MyTextView instance, the signal needs to be disconnected in the
MyTextView destructor.
#include <glib-
object.h>
typedef struct _MyTextView MyTextView;
struct _MyTextView
{
GspellChecker
gulong
*spell_checker;
} word_added_to_personal_handler_id;
;
static void
word_added_to_personal_cb (GspellChecker
*spell_checker, const gchar *word,
gpointe user_data
{ r )
MyTextView *text_view =
user_data;
g_message ("Word ’%s’ has been added to the user’s
personal " "dictionary. text_view=%p will be updated
word,
accordingly.",
text_view
} );
MyTextView *
my_text_view_new (GspellChecker
{
*spell_checker)
MyTextView
*text_view;
g_return_val_if_fail (GSPELL_IS_CHECKER (spell_checker),
NULL);
text_view = g_new0 (MyTextView,
1);
/* We store the spell_checker GObject in the instance
* we increase
variable, so the reference count to be sure that
*
spell_checker during the lifetime of
stays alive
*
text_view.
* Note that spell_checker is provided externally, so
* can live longer than text_view, hence the need to
spell_checker
* signal inthe
disconnect
*
my_text_view_free().
3
/
Most of the time a GTK+ widget doesn’t live longer than the container it is added
and the object listening to the widget signal is usually the container itself. So the widget
dies at the same time as the container, it is not possible for the widget to send a signal while
its container has already been destroyed. In that case, there is thus no point in disconnecting
the signal in the container destructor, since at that point the widget is already freed; and
harder for a dead object to send a signal4 .
4
When I say that it is harder, it is actually impossible, of
54
text_view->spell_checker = g_object_ref
(spell_checker);
text_view->word_added_to_personal_handler_id
= g_signal_connect
(spell_checker, "word-added-to-
G_CALLBACK
personal",
text_view
(word_added_to_personal_cb),
);
return
} text_view;
void
my_text_view_free (MyTextView
{
*text_view)
if (text_view ==
return;
NULL)
if (text_view->spell_checker !=
NULLtext_view->word_added_to_personal_handler_id
&& !
{ = 0)
g_signal_handler_disconnect (text_view-
>spell_checker, text_view-
>word_added_to_personal_handler_id);
/* Here resetting the value to 0 is not
* text_view
necessary will anyway be freed, it is just to
because
* complete
have a more
*
example.
text_view->word_added_to_personal_handler_id
/ =
} 0;
/* The equivalent
* if (text_view-
of:
*
>spell_checker != NULL)
*
{ g_object_unref (text_view-
* text_view->spell_checker
>spell_checker); =
* NULL;
*
}
* After decreasing the reference count, spell_checker
* alive
may ifbe
still another part of the program still
*
references the same
*
spell_checke
g_clear_object
/ (&text_view-
>spell_checker);
g_free
} (text_view);
Listing Disconnecting a signal handler when its
user_dat argument is freed.
a
There are actually other g_signal_handler* functions that permit to discon-
nect signal handlers: ()
• g_signal_handlers_disconnect_by_dat
a()
• g_signal_handlers_disconnect_by_fun
c()
• g_signal_handlers_disconnect_matched
()
It would have been possible to use one of the above functions in Listing 4.5, and
it would have avoided the need to store word_added_to_personal_handler_i .
d
55
The basic g_signal_handler_disconnec function has been used for learning
purposes. t()
Note also that if MyTextView was a GObject class, it would have been possible
to connect to the spell checker signal with g_signal_connect_objec , and
would have removed completely the need tot() manually disconnecting the signal
handler in the MyTextView destructor. One more (small) reason to learn how to
create GObject sub-classes.
4.5. Propertie
If you have looked at the GTK+ or GIO API reference, you must have
that some GObject classes have one or more properties. A property is like
instance variable, also called “attribute”, but is different in the GObject context
5
because it has additional interesting properties .
As previously said at the beginning of section 4.5 p. 50, a good example of a
property is the state of a check button, i.e.a boolean value describing whether
the button is currently checked or not. If you already know a little
you might have found that a check button is available with the GtkCheckButton
widget. But there was a little trap you wanted to find the property that
I was talking about, because the property is implemented in the parent class
GtkToggleButton: the GtkToggleButton:acti property6 .
ve
Th "notify" Signa
7
The main of a property — besides representing a value — is that the
GObject class emits the GObject::noti signal when the value of a property
changes. fy
There is one concept of GObject signals that is not yet explained and is used with
the GObject::noti signal: when emitting a signal, a can be provided.
In the
fy case of the notify signal, the detail is the name of the property whose
value has changed. Since there is only one signal for all properties, thanks to
the detail it is possible to connect a callback to be notified only when a certain
property has changed.If the is not provided when connecting the callback,
the callback would be called when any of the object properties change, which is
generally not what is wanted.
It will be clearer with an example.The Listing 4.6 p. 57 shows how to connect
to the notify signal.Note that instead of connecting to the "notify::act de-
tailed signal, it is actually more convenient to use the ive"
GtkToggleButton::toggle
signal. There are better real-world use-cases where it d is needed to connect to the
notify signal, but at least the Listing 4.6 is hopefully understandable with only
limited GTK+ knowledge (and if you look at the documentation in Devhelp
parallel).
5
Pun intended.
6
In the same way as signals are documented with GTK- as
“ClassName::signal- ”, properties are documented as “ClassName:property-name”.
7
namePun also intended.
56
/* If you look at the notify signal documentation, the
* hasparameter
first the type GObject, not GtkCheckButton. Since
* sub-class of GObject,
GtkCheckButton is a the C language allows to write
*
GtkCheckButton
*
directly
static
/ void
check_button_notify_cb (GtkCheckButton
*check_button, GParamSpec *pspec
gpointe ,user_data
{ r )
/* Called each time that any property of check_button
} changes. */
static void
check_button_notify_active_cb (GtkCheckButton
*check_button, GParamSpec *pspec
gpointe ,user_data
{ r )
MyWindow *window = MY_WINDOW (user_data);
gboolean
active;
active = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON
gtk_widget_set_visible (window->side_panel,
(check_button));
} active);
static GtkWidget *
create_check_button (MyWindow *window)
{
GtkWidget
*check_button;
check_button = gtk_check_button_new_with_label ("Show side
panel");
/* Connect without the
g_signal_connect
detail. */
(check_button, "notify
G_CALLBACK
",
NULL)
(check_button_notify_cb),
;
/* Connect with the detail, to be notified
* the
only GtkToggleButton:active property
when
*
changes.
g_signal_connect
/
(check_button, "notify::acti
G_CALLBACK
ve",
window);
(check_button_notify_active_cb),
return
} check_button;
Listing Connecting to the notify signal to listen to property changes.
57
Property
Another useful aspect of properties is that two properties can easily be bound:
when one property changes,the other is updated to have the same value. Th
same can be accomplished with the "notif signal, but higher-level functions
exist. y"
static GtkWidget *
create_check_button (MyWindow *window)
{
GtkWidget
*check_button;
check_button = gtk_check_button_new_with_label ("Show side
panel");
/* When the GtkToggleButton:active property of check_button
* the GtkWidget:visible property of window->side_panel is
changes,
* have the
updated to same boolean
*
value.
* It would be useful to add G_BINDING_SYNC_CREATE to the
* in that
flags, butcase the code would not be equivalent to
*
thecode
previous
*
Listing.
g_object_bind_property
/ (check_button,
"active", window->side_panel,
G_BINDING_DEFAULT);
"visible",
return
} check_button;
Listing Binding two properties.
58
Part
GT
59
Chapter
Example of a
Application Code
Although not represented on the schema, the entry point of a GTK+ application
– as for every C program – is the
main( function. To create a GTK+
the principal thing to do in main(
) is to create a GtkApplicatio instance, or a
subclass of it.In the schema)we see that
GeditApp isn a subclass ofGtkApplicatio ,
so the main( function of gedit creates aGeditApp object. n
)
1 https://wiki.gnome.org/Apps/
2 https://wiki.gnome.org/Projects/
Gedit
GtkSourceView
60
Figure 5.1: Simplified code architecture of the gedit text editor
GtkApplicatio is the class that contains and represents the whole application.
There is usually only one instance of GtkApplicatio per process, so it can be
n
considered a singleton class. Wha GtkApplicatio
n contains are the windows,
for example the GeditWindow’s in casen of gedit, plus other types of windows like
dialog windows.
We already saw the GtkApplicatio class hierarchy in section 4.1 p. 47 when
explaining OOP inheritance
n with
GObject
GApplicatio
n GtkApplicatio
n
GApplicatio is part of the GIO library and implements the features that are
not related to the Graphical User Interface
n So for a program that runs
in the terminal, it is possible to use
GApplicatio only.
n
An important feature that GApplicatio provides is process uniqueness (but it
can be disabled if not wanted).
n What process uniqueness does is to have only one
process per application per user session.For that feature to work, an application
ID must be provided when creating the GApplicatio object. With that
GApplicatio looks if another process alreadyn runs the same application in the
same user session; if it is the case, it communicates to the primary instance the
n
actions that need to be done (for example opening a new window, or opening a
new file in an existing window, etc).When the actions are done on the primary
instance, the second process exits immediately.On GApplicatio uses the
D-Bus Inter-Process Communication (IPC) system to communicate
n between the
two processes.
61
Process uniqueness has several advantages, to give a few concrete examples:
• For an application with a tabbed document interface, when clicking on
a file in a file manager like Nautilus,the file can be opened in a new tab
instead of creating each time a new window. For this to work, inter-process
communication is needed in one form or another;
• An application doesn’t need to synchronize explicitly its state and data
between different processes. For the sake of argument, let’s say that
gedit the user can create custom “build tools”, to compile the current file
or project. gedit saves the custom build tools in an XML file and are shown
in the menu to execute their commands.On Linux, the XML file is
for example in the user’s ~/.local/share/ Without process
uniqueness, if one gedit process modifies the custom build tools, the other
gedit processes need to reload the XML file, and need to ensure that there
are no races (two different gedit processes must not modify the XML file at
the same time). With process uniqueness,that problem doesn’t exist,al
the gedit windows share the same application state, and the developer can
assume that only one process per user can modify the XML 3file (of course
the user has still the possibility to edit the XML file by hand, but in
case the application can just be restarted, normally the user is expected to
modify the build tools from the GUI that gedit provides).
Another important feature ofGApplicatio is to run the main event loop. Th
GLib main event loop was described
n in section 2.3 p. 31. Wit GApplicatio ,
this is done with the g_application_ru function. A minimalistic
n version of
the main( function in gedit would look like:
n()
int )
main (int argc
char ,
{ **argv)
GeditApp
int
*app;
status;
/* Init i18n
(internationalization) here. */
app = gedit_app_new
status = g_application_run (G_APPLICATION (app), argc,
();
g_object_unref
argv);
(app);
return
} status;
Wha GeditApp does is basically what would need to be done inmain( if there
was no GtkApplicatio subclass. This includes: )
3
n
Note that this would not be true if it was possible to open severalgraphical sessions for the
same user, on the same machine (with multi-seat support) or at least sharing the backing storage
for the home directory (for example with NFS mounts). But GNOME and most
don’t support this, a user can open at most one graphical session at a time for the same home
directory. For logins on the same physical machine, this is enforced by GDM (the
display manager and login screen) and D-Bus. For NFS mounts this is not enforced, but
the same user opens several graphical sessions on different computers, some programs might
misbehave. So although the GApplicatio process uniqueness is documented as being per
user session, in practice we can
n say that it is simply per
62
• Configuring the GtkApplicatio object correctly, for example giving the
application n
4
• Connecting callbacks to some signals , for example to create aGeditWindow
when needed;
• Implementing application-wide GAction’s GAction is a class part of
that represents an action that the user can trigger. An application-wide
action is for example to quit the application, or to open the preferences
dialog (because the preferences are applied to the whole application).
When you start writing a new GTK+ you don’t see directly the
need for a GtkApplicatio subclass, since the code inmain( , plus the callbacks,
are small.
n But when more and more features are ) added,it is a good idea
at some point to move the code to a GtkApplicatio subclass. Or to create a
subclass directly. A subclass is especially
n useful when the need arises to store
additional data.
5. GeditWindo
In the schema, the “1” and “1..*” notation means that GeditApp object
contains one or severalGeditWindow objects, and that aGeditWindow is contained
in exactly one GeditApp ( GeditWindow cannot be contained in severalGeditApp
objects, there is anyway only oneGeditApp instance per process).
GeditWindow is responsible to create the main UI, creating other widgets and as-
sembling them in aGtkGri container for example.Another thing thatGeditWindow
does is to implement
d the GAction that have an effect only on the current win-
dow, for example an action
s to close the window, or save the current document.
When implementing a GAction, GeditWindow can of course delegate most of its
work to other classes contained inGeditWindow.
At the top of a main application window,there is usually a GtkHeaderBar,
shows the window title, some buttons and an “hamburger” menu. Alternatively
an application can have a traditional menubar and toolbar.
Besides the headerbar, GeditWindow creates a GeditStatusba widget and adds
it to the bottom of the window. It also creates
r two GeditPanel , one on the
left side of the window, and the other on the bottom,s above the
GeditStatusba .
Each panel can contain several elements. For example the sider panel contains
an integrated file browser, and the bottom panel can contain a terminal, among
other things5 .
4
But note that in a GObject subclass, instead of connecting callbacks to signals of a parent
class with e.g. g_signal_connect , it is better to override the virtual functions instead.
5
The current()gedit code actually doesn’t contain a GeditPane class anymore, but it was
the case in an earlier version. Adding GeditPane to the l diagram was done to show a possible
implementation of panels in an application.
l If your application contains only one element in a
panel, no need to have a Pane class, you can directly add the element to the window.
l
63
GeditWindow also creates a GeditNotebook, the main part of the window.
What OOP is about is to pack data and behavior together, and delegate
some of the work to other classes.Class inheritance makes sense when we want
to add more behavior to an existing class, with possible additional data re-
lated to the added behavior. GeditVie is a subclass of GtkSourceView because
GeditVie is GtkSourceView; wthat GeditVie operates on the same base data
as GtkSourceView. In addition, it permits
w w GeditTa to delegate some of its
b
64
work, with the goal to have smaller, more manageable classes. Smaller in two
ways: less code, and less instance variables.
So, during the lifetime of a GTK+ the programmer often needs to
refactor the code, creating new classes, delegating more work.
The opposite can
happen when application code is moved to the underlying library; for example, if
all the features of GeditVie are added to the GtkSourceView class; in that case,
the GeditVie subclass
w doesn’t make sense anymore.
w
5. Composite
Composite widgets are containers that already contain a useful collection of child
widgets in a nice package. Implementing a composite widget is easy6 , you just
need to:
1. Subclass a container likeGtkGri or GtkBin or GtkWindow;
d
2. In the constructor of the class,create the child widgets and add them to
the container.
In the gedit class schema, the composite widgets are the subclasses ofGtkGri
(GeditPane and GeditTa ) GeditWindow. d
l b
GeditWindow is an indirect subclass ofGtkBin, so it can contain at most one child
widget. That’s GeditWindow uses a GtkGri as its child widget, so that the
GtkGri can contain in turn all the window
d elements.
d
By default GeditTa has only one child widget, the GtkScrolledWindow that
contains the GeditVie
b . Bu GeditTa has a function to add aGtkInfoBa at the
top, showingwfor example anb error message. r
So, while GtkGri is a general-purpose container that doesn’t contain any child
widget initially,
d a composite widget is a specialized container that already con-
tains specific child widgets.Writing composite widgets are a convenient way to
code applications.
6
Once you know how to subclass a GObject class.
65
Part
Further
66
Chapter
Further
At this point you should know the basics of GLib core and You
need to know everything about GLib core and GObject to continue, but having
at least a basic understanding will allow you to more easily learn GTK+
GIO, or any other GObject-based library for that
6. GTK+ and
67
There is also a series of small tutorials on various GLib/GTK+
https://wiki.gnome.org/
HowDoI
The chapter 4 explained how to use an existing GObject class, which is very
useful to learn but it didn’t explain how to create your own
classes. Writing your own GObject classes permit to have reference counting,
you can create your own properties and signals,you can implement interfaces,
overriding virtual functions (if the virtual function is not associated to a signal),
etc.
As was explained at the beginning of chapter 4, if you want to learn more in-
depth information on GObject and to know how to create sub-classes, the GOb-
ject reference documentation contains introductory chapters: “Concepts”
“Tutorial”, available as usual in Devhelp or online
https://developer.gnome.org/gobject/
stable/
6. Build
6.3. The
The Autotools comprise three main components: Autoconf, Automake and Libtool.
It is based on shell scripts, m4 macros and make.
Macros are available for various purposes (the user documentation, code coverage
statistics for unit tests, The most recent book on the subject is Autotools,
by John Calcote
But the Autotools have the reputation to be difficult to learn.
6.3. Meso
Meson is a fairly new build system, is easier to learn than the Autotools, and
also results to faster builds.Some GNOME modules already use Meson.See the
website for more information:
http://
mesonbuild.com/
68
6. Programming Best-
1
Although the editor of Code Complete is Microsoft Press, the book is not related to Mi-
crosoft or Windows. The author sometimes explains stuff related to open source, UNIX
Linux but one can regret the total absence of the mention “free/libre software” and all the
benefits of freedom, in particular for this kind of book: being able to learn by reading other’s
code. But if you are here, you hopefully already know all of this.
69
Bibliograph
[5] Paul Abrahams, UNIX for the Impatient, Second Edition, Addison-
1995.
[9] GTK-Doc
https://developer.gnome.org/gtk-doc-
manual/
[10] GObject
https://wiki.gnome.org/Projects/
GObjectIntrospection
[11] GNOME Programming
https://developer.gnome.org/programming-guidelines/
stable/
70