Python GTK 3 Tutorial
Python GTK 3 Tutorial
Release 3.4
Sebastian Plsterl
Deprecated
Installation
1.1 Dependencies . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1.2 Prebuilt Packages . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1.3 Installing From Source . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
3
3
3
3
Getting Started
2.1 Simple Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
2.2 Extended Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
5
5
6
Basics
3.1 Main loop and Signals . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
3.2 Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
9
9
10
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
11
11
11
13
13
Layout Containers
5.1 Boxes . . . . . . . . . .
5.2 Grid . . . . . . . . . . .
5.3 ListBox . . . . . . . . .
5.4 Stack and StackSwitcher
5.5 HeaderBar . . . . . . .
5.6 FlowBox . . . . . . . .
5.7 Notebook . . . . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
15
15
16
17
20
21
22
25
Label
6.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
27
28
Entry
7.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
31
31
Button Widgets
8.1 Button . . .
8.2 ToggleButton
8.3 CheckButton
8.4 RadioButton
35
35
36
37
37
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
8.5
8.6
8.7
LinkButton . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
SpinButton . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
Switch . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
38
39
41
ProgressBar
9.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
43
43
10 Spinner
10.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
47
47
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
49
49
51
51
52
53
12 CellRenderers
12.1 CellRendererText . . .
12.2 CellRendererToggle .
12.3 CellRendererPixbuf .
12.4 CellRendererCombo .
12.5 CellRendererProgress
12.6 CellRendererSpin . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
57
57
59
60
62
63
65
13 ComboBox
13.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
69
69
14 IconView
14.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
73
74
.
.
.
.
77
77
77
78
79
16 Dialogs
16.1 Custom Dialogs . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
16.2 MessageDialog . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
16.3 FileChooserDialog . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
85
85
87
89
17 Clipboard
17.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
93
93
95
95
96
96
96
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
99
. 99
. 100
. 100
. 101
ii
.
.
.
.
.
.
.
.
20 Objects
20.1 Inherit from GObject.GObject
20.2 Signals . . . . . . . . . . . .
20.3 Properties . . . . . . . . . . .
20.4 API . . . . . . . . . . . . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
103
103
103
104
107
21 Application
21.1 Actions . . . .
21.2 Menus . . . .
21.3 Command Line
21.4 Example . . .
21.5 See Also . . .
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
109
109
109
109
110
113
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
.
22 Menus
115
22.1 Actions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 115
22.2 UI Manager . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 116
22.3 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 116
23 Table
121
23.1 Example . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 121
24 Indices and tables
123
iii
iv
Release 3.4
Date December 04, 2016
Copyright GNU Free Documentation License 1.3 with no Invariant Sections, no Front-Cover Texts, and
no Back-Cover Texts
This tutorial gives an introduction to writing GTK+ 3 applications in Python.
Prior to working through this tutorial, it is recommended that you have a reasonable grasp of the Python programming
language. GUI programming introduces new problems compared to interacting with the standard output (console /
terminal). It is necessary for you to know how to create and run Python files, understand basic interpreter errors, and
work with strings, integers, floats and Boolean values. For the more advanced widgets in this tutorial, good knowledge
of lists and tuples will be needed.
Although this tutorial describes the most important classes and methods within GTK+ 3, it is not supposed to serve as
an API reference. Please refer to the GTK+ 3 Reference Manual for a detailed description of the API. Also theres a
Python-specific reference available.
Contents:
Deprecated
Deprecated
CHAPTER 1
Installation
The first step before we start with actual coding consists of setting up PyGObject and its dependencies. PyGObject is a
Python module that enables developers to access GObject-based libraries such as GTK+ within Python. It exclusively
supports GTK+ version 3 or later. If you want to use GTK+ 2 in your application, use PyGTK, instead.
1.1 Dependencies
GTK+3
Python 2 (2.6 or later) or Python 3 (3.1 or later)
gobject-introspection
It will print any applications and libraries that are currently missing on your system but required for building. You
should install those using your distributions package repository. A list of package names for different distributions is
maintained on the GNOME wiki. Run the command above again to ensure the required tools are present.
Executing the following command will build PyGObject and all its dependencies:
1
https://download.gnome.org/teams/releng/
Chapter 1. Installation
CHAPTER 2
Getting Started
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
6
7
8
win = Gtk.Window()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
In the beginning, we have to import the Gtk module to be able to access GTK+s classes and functions. Since a
users system can have multiple versions of GTK+ installed at the same, we want to make sure that we when we
import Gtk that it referrs to GTK+ 3 and not any other version of the library, which is the purpose of statement
gi.require_version(Gtk, 3.0).
The next line creates an empty window.
win = Gtk.Window()
Followed by connecting to the windows delete event to ensure that the application is terminated if we click on the x
to close the window.
win.connect("delete-event", Gtk.main_quit)
Finally, we start the GTK+ processing loop which we quit when the window is closed (see line 5).
Gtk.main()
To run the program, open a terminal, change to the directory of the file, and enter:
python simple_example.py
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class MyWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="Hello World")
7
8
9
10
11
12
13
14
15
16
17
win = MyWindow()
18
19
20
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
This example differs from the simple example as we sub-class Gtk.Window to define our own MyWindow class.
class MyWindow(Gtk.Window):
In the classs constructor we have to call the constructor of the super class. In addition, we tell it to set the value of the
property title to Hello World.
Gtk.Window.__init__(self, title="Hello World")
The next three lines are used to create a button widget, connect to its clicked signal and add it as child to the top-level
window.
self.button = Gtk.Button(label="Click Here")
self.button.connect("clicked", self.on_button_clicked)
self.add(self.button)
Accordingly, the method on_button_clicked() will be called if you click on the button.
def on_button_clicked(self, widget):
print("Hello World")
The last block, outside of the class, is very similar to the simple example above, but instead of creating an instance of
the generic Gtk.Window class, we create an instance of MyWindow.
CHAPTER 3
Basics
This section will introduce some of the most important aspects of GTK+.
Firstly, widget is an instance of a widget we created earlier. Next, the event we are interested in. Each widget has its
own particular events which can occur. For instance, if you have a button you usually want to connect to the clicked
event. This means that when the button is clicked, the signal is issued. Thirdly, the callback argument is the name of
the callback function. It contains the code which runs when signals of the specified type are issued. Finally, the data
argument includes any data which should be passed when the signal is issued. However, this argument is completely
optional and can be left out if not required.
The function returns a number that identifies this particular signal-callback pair. It is required to disconnect from a
signal such that the callback function will not be called during any future or currently ongoing emissions of the signal
it has been connected to.
widget.disconnect(handler_id)
If you have lost the handler_id for some reason (for example the handlers were installed using
Gtk.Builder.connect_signals()), you can still disconnect a specific callback using the function
disconnect_by_func():
widget.disconnect_by_func(callback)
Almost all applications will connect to the delete-event signal of the top-level window. It is emitted if a user requests
that a toplevel window is closed. The default handler for this signal destroys the window, but does not terminate the
application. Connecting the delete-event signal to the function Gtk.main_quit() will result in the desired
behaviour.
9
window.connect("delete-event", Gtk.main_quit)
3.2 Properties
Properties describe the configuration and state of widgets. As for signals, each widget has its own particular set of
properties. For example, a button has the property label which contains the text of the label widget inside the button.
You can specify the name and value of any number of properties as keyword arguments when creating an instance of
a widget. To create a label aligned to the right with the text Hello World and an angle of 25 degrees, use:
label = Gtk.Label(label="Hello World", angle=25, halign=Gtk.Align.END)
which is equivalent to
label = Gtk.Label()
label.set_label("Hello World")
label.set_angle(25)
label.set_halign(Gtk.Align.END)
Instead of using getters and setters you can also get and set the gobject properties through the props
property such as widget.props.prop_name = value.
This is equivelent to the more verbose
widget.get_property("prop-name") and widget.set_property("prop-name", value).
To see which properties are available for a widget in the running version of GTK you can dir the props property:
widget = Gtk.Box()
print(dir(widget.props))
This will print in the console the list of properties a Gtk.Box has.
10
Chapter 3. Basics
CHAPTER 4
This section explains how strings are represented in Python 2.x, Python 3.x and GTK+ and discusses common errors
that arise when working with strings.
4.1 Definitions
Conceptionally, a string is a list of characters such as A, B, C or . Characters are abstract representations and
their meaning depends on the language and context they are used in. The Unicode standard describes how characters
are represented by code points. For example the characters above are represented with the code points U+0041,
U+0042, U+0043, and U+00C9, respectively. Basically, code points are numbers in the range from 0 to 0x10FFFF.
As mentioned earlier, the representation of a string as a list of code points is abstract. In order to convert this abstract
representation into a sequence of bytes the Unicode string must be encoded. The simplest form of encoding is ASCII
and is performed as follows:
1. If the code point is < 128, each byte is the same as the value of the code point.
2. If the code point is 128 or greater, the Unicode string cant be represented in this encoding. (Python raises a
UnicodeEncodeError exception in this case.)
Although ASCII encoding is simple to apply it can only encode for 128 different characters which is hardly enough.
One of the most commonly used encodings that addresses this problem is UTF-8 (it can handle any Unicode code
point). UTF stands for Unicode Transformation Format, and the 8 means that 8-bit numbers are used in the
encoding.
4.2 Python 2
4.2.1 Python 2.xs Unicode Support
Python 2 comes with two different kinds of objects that can be used to represent strings, str and unicode. Instances
of the latter are used to express Unicode strings, whereas instances of the str type are byte representations (the
encoded string). Under the hood, Python represents Unicode strings as either 16- or 32-bit integers, depending on how
the Python interpreter was compiled. Unicode strings can be converted to 8-bit strings with unicode.encode():
>>> unicode_string = u"Fu\u00dfb\u00e4lle"
>>> print unicode_string
Fublle
>>> type(unicode_string)
<type 'unicode'>
11
>>> unicode_string.encode("utf-8")
'Fu\xc3\x9fb\xc3\xa4lle'
Pythons 8-bit strings have a str.decode() method that interprets the string using the given encoding:
>>> utf8_string = unicode_string.encode("utf-8")
>>> type(utf8_string)
<type 'str'>
>>> u2 = utf8_string.decode("utf-8")
>>> unicode_string == u2
True
Unfortunately, Python 2.x allows you to mix unicode and str if the 8-bit string happened to contain only 7-bit
(ASCII) bytes, but would get UnicodeDecodeError if it contained non-ASCII values:
>>> utf8_string = " sind rund"
>>> unicode_string + utf8_string
u'Fu\xdfb\xe4lle sind rund'
>>> utf8_string = " k\xc3\xb6nnten rund sein"
>>> print utf8_string
knnten rund sein
>>> unicode_string + utf8_string
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
UnicodeDecodeError: 'ascii' codec can't decode byte 0xc3 in position 2:
ordinal not in range(128)
Note the warning at the end. Although we called Gtk.Label.set_text() with a unicode instance as argument,
Gtk.Label.get_text() will always return a str instance. Accordingly, txt and unicode_string are not
equal.
This is especially important if you want to internationalize your program using gettext. You have to make sure that
gettext will return UTF-8 encoded 8-bit strings for all languages. In general it is recommended to not use unicode
objects in GTK+ applications at all and only use UTF-8 encoded str objects since GTK+ does not fully integrate
with unicode objects. Otherwise, you would have to decode the return values to Unicode strings each time you call
a GTK+ method:
12
4.3 Python 3
4.3.1 Python 3.xs Unicode support
Since Python 3.0, all strings are stored as Unicode in an instance of the str type. Encoded strings on the other hand
are represented as binary data in the form of instances of the bytes type. Conceptionally, str refers to text, whereas
bytes refers to data. Use str.encode() to go from str to bytes, and bytes.decode() to go from bytes
to str.
In addition, it is no longer possible to mix Unicode strings with encoded strings, because it will result in a
TypeError:
>>> text = "Fu\u00dfb\u00e4lle"
>>> data = b" sind rund"
>>> text + data
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Can't convert 'bytes' object to str implicitly
>>> text + data.decode("utf-8")
'Fublle sind rund'
>>> text.encode("utf-8") + data
b'Fu\xc3\x9fb\xc3\xa4lle sind rund'
4.4 References
Whats new in Python 3.0 describes the new concepts that clearly distinguish between text and data.
The Unicode HOWTO discusses Python 2.xs support for Unicode, and explains various problems that people commonly encounter when trying to work with Unicode.
The Unicode HOWTO for Python 3.x discusses Unicode support in Python 3.x.
4.3. Python 3
13
UTF-8 encoding table and Unicode characters contains a list of Unicode code points and their respective UTF-8
encoding.
14
CHAPTER 5
Layout Containers
While many GUI toolkits require you to precisely place widgets in a window, using absolute positioning, GTK+ uses
a different approach. Rather than specifying the position and size of each widget in the window, you can arrange your
widgets in rows, columns, and/or tables. The size of your window can be determined automatically, based on the sizes
of the widgets it contains. And the sizes of the widgets are, in turn, determined by the amount of text they contain, or
the minimum and maximum sizes that you specify, and/or how you have requested that the available space should be
shared between sets of widgets. You can perfect your layout by specifying padding distance and centering values for
each of your widgets. GTK+ then uses all this information to resize and reposition everything sensibly and smoothly
when the user manipulates the window.
GTK+ arranges widgets hierarchically, using containers. They are invisible to the end user and are inserted into a window, or placed within each other to layout components. There are two flavours of containers: single-child containers,
which are all descendants of Gtk.Bin, and multiple-child containers, which are descendants of Gtk.Container.
The most commonly used are vertical or horizontal boxes (Gtk.Box) and grids (Gtk.Grid).
5.1 Boxes
Boxes are invisible containers into which we can pack our widgets. When packing widgets into a horizontal box, the
objects are inserted horizontally from left to right or right to left depending on whether Gtk.Box.pack_start()
or Gtk.Box.pack_end() is used. In a vertical box, widgets are packed from top to bottom or vice versa. You may
use any combination of boxes inside or beside other boxes to create the desired effect.
5.1.1 Example
Lets take a look at a slightly modified version of the extended example with two buttons.
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class MyWindow(Gtk.Window):
15
def __init__(self):
Gtk.Window.__init__(self, title="Hello World")
7
8
9
self.box = Gtk.Box(spacing=6)
self.add(self.box)
10
11
12
self.button1 = Gtk.Button(label="Hello")
self.button1.connect("clicked", self.on_button1_clicked)
self.box.pack_start(self.button1, True, True, 0)
13
14
15
16
self.button2 = Gtk.Button(label="Goodbye")
self.button2.connect("clicked", self.on_button2_clicked)
self.box.pack_start(self.button2, True, True, 0)
17
18
19
20
21
22
23
24
25
26
27
28
29
30
win = MyWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
First, we create a horizontally orientated box container where 6 pixels are placed between children. This box becomes
the child of the top-level window.
self.box = Gtk.Box(spacing=6)
self.add(self.box)
While with Gtk.Box.pack_start() widgets are positioned from left to right, Gtk.Box.pack_end() positions them from right to left.
5.2 Grid
Gtk.Grid is a container which arranges its child widgets in rows and columns, but you do not need to specify the
dimensions in the constructor. Children are added using Gtk.Grid.attach(). They can span multiple rows or
columns. It is also possible to add a child next to an existing child, using Gtk.Grid.attach_next_to().
Gtk.Grid can be used like a Gtk.Box by just using Gtk.Grid.add(), which will place children next to each
other in the direction determined by the orientation property (defaults to Gtk.Orientation.HORIZONTAL).
16
5.2.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class GridWindow(Gtk.Window):
6
7
8
def __init__(self):
Gtk.Window.__init__(self, title="Grid Example")
9
10
11
grid = Gtk.Grid()
self.add(grid)
12
13
14
15
16
17
18
button1
button2
button3
button4
button5
button6
=
=
=
=
=
=
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
1")
2")
3")
4")
5")
6")
19
20
21
22
23
24
25
grid.add(button1)
grid.attach(button2, 1, 0, 2, 1)
grid.attach_next_to(button3, button1, Gtk.PositionType.BOTTOM, 1, 2)
grid.attach_next_to(button4, button3, Gtk.PositionType.RIGHT, 2, 1)
grid.attach(button5, 1, 2, 1, 1)
grid.attach_next_to(button6, button5, Gtk.PositionType.RIGHT, 1, 1)
26
27
28
29
30
win = GridWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
5.3 ListBox
A Gtk.ListBox is a vertical container that contains Gtk.ListBoxRow children. These rows can by dynamically
sorted and filtered, and headers can be added dynamically depending on the row content. It also allows keyboard and
mouse navigation and selection like a typical list.
Using Gtk.ListBox is often an alternative to Gtk.TreeView, especially when the list contents has a more complicated layout than what is allowed by a Gtk.CellRenderer, or when the contents is interactive (i.e. has a button
in it).
5.3. ListBox
17
Although a Gtk.ListBox must have only Gtk.ListBoxRow children you can add any kind of widget to it via
Gtk.Container.add(), and a Gtk.ListBoxRow widget will automatically be inserted between the list and
the widget.
5.3.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
6
7
8
9
class ListBoxRowWithData(Gtk.ListBoxRow):
def __init__(self, data):
super(Gtk.ListBoxRow, self).__init__()
self.data = data
self.add(Gtk.Label(data))
10
11
class ListBoxWindow(Gtk.Window):
12
def __init__(self):
Gtk.Window.__init__(self, title="ListBox Demo")
self.set_border_width(10)
13
14
15
16
17
18
19
listbox = Gtk.ListBox()
listbox.set_selection_mode(Gtk.SelectionMode.NONE)
box_outer.pack_start(listbox, True, True, 0)
20
21
22
23
row = Gtk.ListBoxRow()
hbox = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=50)
row.add(hbox)
24
25
26
18
27
28
vbox = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
hbox.pack_start(vbox, True, True, 0)
29
30
31
32
33
34
35
36
37
switch = Gtk.Switch()
switch.props.valign = Gtk.Align.CENTER
hbox.pack_start(switch, False, True, 0)
38
39
listbox.add(row)
40
41
42
43
44
45
46
47
row = Gtk.ListBoxRow()
hbox = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=50)
row.add(hbox)
label = Gtk.Label("Enable Automatic Update", xalign=0)
check = Gtk.CheckButton()
hbox.pack_start(label, True, True, 0)
hbox.pack_start(check, False, True, 0)
48
49
listbox.add(row)
50
51
52
53
54
55
56
57
58
59
row = Gtk.ListBoxRow()
hbox = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=50)
row.add(hbox)
label = Gtk.Label("Date Format", xalign=0)
combo = Gtk.ComboBoxText()
combo.insert(0, "0", "24-hour")
combo.insert(1, "1", "AM/PM")
hbox.pack_start(label, True, True, 0)
hbox.pack_start(combo, False, True, 0)
60
61
listbox.add(row)
62
63
64
listbox_2 = Gtk.ListBox()
items = 'This is a sorted ListBox Fail'.split()
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
win = ListBoxWindow()
win.connect("delete-event", Gtk.main_quit)
5.3. ListBox
19
85
86
win.show_all()
Gtk.main()
5.4.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class StackWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="Stack Demo")
self.set_border_width(10)
7
8
9
10
11
12
13
stack = Gtk.Stack()
stack.set_transition_type(Gtk.StackTransitionType.SLIDE_LEFT_RIGHT)
stack.set_transition_duration(1000)
14
15
16
17
18
19
20
20
21
22
23
label = Gtk.Label()
label.set_markup("<big>A fancy label</big>")
stack.add_titled(label, "label", "A label")
24
25
26
27
28
stack_switcher = Gtk.StackSwitcher()
stack_switcher.set_stack(stack)
vbox.pack_start(stack_switcher, True, True, 0)
vbox.pack_start(stack, True, True, 0)
29
30
31
32
33
win = StackWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
5.5 HeaderBar
A Gtk.HeaderBar is similar to a horizontal Gtk.Box, it allows to place children at the start or the end. In addition,
it allows a title to be displayed. The title will be centered with respect to the width of the box, even if the children at
either side take up different amounts of space.
Since GTK+ now supports Client Side Decoration, a Gtk.HeaderBar can be used in place of the title bar (which is
rendered by the Window Manager).
A Gtk.HeaderBar is usually located across the top of a window and should contain commonly used controls which
affect the content below. They also provide access to window controls, including the close window button and window
menu.
5.5.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Gio
4
5
class HeaderBarWindow(Gtk.Window):
6
7
8
9
10
def __init__(self):
Gtk.Window.__init__(self, title="HeaderBar Demo")
self.set_border_width(10)
self.set_default_size(400, 200)
5.5. HeaderBar
21
11
hb = Gtk.HeaderBar()
hb.set_show_close_button(True)
hb.props.title = "HeaderBar example"
self.set_titlebar(hb)
12
13
14
15
16
button = Gtk.Button()
icon = Gio.ThemedIcon(name="mail-send-receive-symbolic")
image = Gtk.Image.new_from_gicon(icon, Gtk.IconSize.BUTTON)
button.add(image)
hb.pack_end(button)
17
18
19
20
21
22
box = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL)
Gtk.StyleContext.add_class(box.get_style_context(), "linked")
23
24
25
button = Gtk.Button()
button.add(Gtk.Arrow(Gtk.ArrowType.LEFT, Gtk.ShadowType.NONE))
box.add(button)
26
27
28
29
button = Gtk.Button()
button.add(Gtk.Arrow(Gtk.ArrowType.RIGHT, Gtk.ShadowType.NONE))
box.add(button)
30
31
32
33
hb.pack_start(box)
34
35
self.add(Gtk.TextView())
36
37
38
39
40
41
win = HeaderBarWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
42
5.6 FlowBox
A Gtk.FlowBox is a container that positions child widgets in sequence according to its orientation.
For instance, with the horizontal orientation, the widgets will be arranged from left to right, starting a new row under
the previous row when necessary. Reducing the width in this case will require more rows, so a larger height will be
requested.
Likewise, with the vertical orientation, the widgets will be arranged from top to bottom, starting a new column to the
right when necessary. Reducing the height will require more columns, so a larger width will be requested.
The children of a Gtk.FlowBox can be dynamically sorted and filtered.
Although a Gtk.FlowBox must have only Gtk.FlowBoxChild children, you can add any kind of widget to it
via Gtk.Container.add(), and a Gtk.FlowBoxChild widget will automatically be inserted between the box
and the widget.
22
5.6.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Gdk
4
5
class FlowBoxWindow(Gtk.Window):
6
7
8
9
10
def __init__(self):
Gtk.Window.__init__(self, title="FlowBox Demo")
self.set_border_width(10)
self.set_default_size(300, 250)
11
12
13
14
15
16
self.set_titlebar(header)
17
18
19
scrolled = Gtk.ScrolledWindow()
scrolled.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC)
20
21
22
23
24
flowbox = Gtk.FlowBox()
flowbox.set_valign(Gtk.Align.START)
flowbox.set_max_children_per_line(30)
flowbox.set_selection_mode(Gtk.SelectionMode.NONE)
25
26
self.create_flowbox(flowbox)
27
28
scrolled.add(flowbox)
29
30
31
self.add(scrolled)
self.show_all()
32
33
34
35
36
rgba = Gdk.RGBA.from_color(color)
5.6. FlowBox
23
button = Gtk.Button()
37
38
area = Gtk.DrawingArea()
area.set_size_request(24, 24)
area.override_background_color(0, rgba)
39
40
41
42
button.add(area)
43
44
return button
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
24
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
'chartreuse',
'chartreuse1',
'chartreuse2',
'chartreuse3',
'chartreuse4',
'chocolate',
'chocolate1',
'chocolate2',
'chocolate3',
'chocolate4',
'coral',
'coral1',
'coral2',
'coral3',
'coral4'
]
111
112
113
114
115
116
117
118
119
120
win = FlowBoxWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
5.7 Notebook
The Gtk.Notebook widget is a Gtk.Container whose children are pages that can be switched between using
tab labels along one edge.
There are many configuration options for GtkNotebook. Among other things, you can choose on which edge
the tabs appear (see Gtk.Notebook.set_tab_pos()), whether, if there are too many tabs to fit the notebook should be made bigger or scrolling arrows added (see Gtk.Notebook.set_scrollable(), and whether
there will be a popup menu allowing the users to switch pages (see Gtk.Notebook.popup_enable(),
Gtk.Notebook.popup_disable()).
5.7.1 Example
1
2
import gi
gi.require_version('Gtk', '3.0')
5.7. Notebook
25
4
5
class MyWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="Simple Notebook Example")
self.set_border_width(3)
7
8
9
10
self.notebook = Gtk.Notebook()
self.add(self.notebook)
11
12
13
self.page1 = Gtk.Box()
self.page1.set_border_width(10)
self.page1.add(Gtk.Label('Default Page!'))
self.notebook.append_page(self.page1, Gtk.Label('Plain Title'))
14
15
16
17
18
self.page2 = Gtk.Box()
self.page2.set_border_width(10)
self.page2.add(Gtk.Label('A page with an image for a Title.'))
self.notebook.append_page(
self.page2,
Gtk.Image.new_from_icon_name(
"help-about",
Gtk.IconSize.MENU
)
)
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
win = MyWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
26
CHAPTER 6
Label
Labels are the main method of placing non-editable text in windows, for instance to place a title next to a
Gtk.Entry widget. You can specify the text in the constructor, or later with the Gtk.Label.set_text()
or Gtk.Label.set_markup() methods.
The width of the label will be adjusted automatically. You can produce multi-line labels by putting line breaks (\n)
in the label string.
Labels can be made selectable with Gtk.Label.set_selectable(). Selectable labels allow the user to copy
the label contents to the clipboard. Only labels that contain useful-to-copy information such as error messages
should be made selectable.
The label text can be justified using the Gtk.Label.set_justify() method. The widget is also capable of
word-wrapping, which can be activated with Gtk.Label.set_line_wrap().
Gtk.Label support some simple formatting, for instance allowing you to make some text bold, colored, or larger.
You can do this by providing a string to Gtk.Label.set_markup(), using the Pango Markup syntax 1 . For
instance, <b>bold text</b> and <s>strikethrough text</s>. In addition, Gtk.Label supports
clickable hyperlinks. The markup for links is borrowed from HTML, using the a with href and title attributes. GTK+
renders links similar to the way they appear in web browsers, with colored, underlined text. The title attribute is
displayed as a tooltip on the link.
label.set_markup("Go to <a href=\"http://www.gtk.org\" "
"title=\"Our website\">GTK+ website</a> for more")
Labels may contain mnemonics. Mnemonics are underlined characters in the label, used for keyboard navigation.
Mnemonics are created by providing a string with an underscore before the mnemonic character, such as _File,
to the functions Gtk.Label.new_with_mnemonic() or Gtk.Label.set_text_with_mnemonic().
Mnemonics automatically activate any activatable widget the label is inside, such as a Gtk.Button; if
the label is not inside the mnemonics target widget, you have to tell the label about the target using
Gtk.Label.set_mnemonic_widget().
1
27
6.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class LabelWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="Label Example")
7
8
9
hbox = Gtk.Box(spacing=10)
hbox.set_homogeneous(False)
vbox_left = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=10)
vbox_left.set_homogeneous(False)
vbox_right = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, spacing=10)
vbox_right.set_homogeneous(False)
10
11
12
13
14
15
16
17
18
19
20
21
28
Chapter 6. Label
22
23
24
25
26
label = Gtk.Label()
label.set_text("This is a left-justified label.\nWith multiple lines.")
label.set_justify(Gtk.Justification.LEFT)
vbox_left.pack_start(label, True, True, 0)
27
28
29
30
31
label = Gtk.Label(
"This is a right-justified label.\nWith multiple lines.")
label.set_justify(Gtk.Justification.RIGHT)
vbox_left.pack_start(label, True, True, 0)
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
label = Gtk.Label()
label.set_markup("Text can be <small>small</small>, <big>big</big>, "
"<b>bold</b>, <i>italic</i> and even point to "
"somewhere in the <a href=\"http://www.gtk.org\" "
"title=\"Click to find out more\">internets</a>.")
label.set_line_wrap(True)
vbox_left.pack_start(label, True, True, 0)
64
65
66
67
68
label = Gtk.Label.new_with_mnemonic(
"_Press Alt + P to select button to the right")
vbox_left.pack_start(label, True, True, 0)
label.set_selectable(True)
69
70
71
72
73
74
self.add(hbox)
75
76
77
78
79
window = LabelWindow()
window.connect("delete-event", Gtk.main_quit)
window.show_all()
Gtk.main()
6.1. Example
29
30
Chapter 6. Label
CHAPTER 7
Entry
Entry widgets allow the user to enter text. You can change the contents with the Gtk.Entry.set_text() method,
and read the current contents with the Gtk.Entry.get_text() method. You can also limit the number of characters the Entry can take by calling Gtk.Entry.set_max_length().
Occasionally you might want to make an Entry widget read-only. This can be done by passing False to the
Gtk.Entry.set_editable() method.
Entry widgets can also be used to retrieve passwords from the user. It is common practice to hide the characters typed
into the entry to prevent revealing the password to a third party. Calling Gtk.Entry.set_visibility() with
False will cause the text to be hidden.
Gtk.Entry has the ability to display progress or activity information behind the text. This is similar to
Gtk.ProgressBar widget and is commonly found in web browsers to indicate how much of a page download has
been completed. To make an entry display such information, use Gtk.Entry.set_progress_fraction(),
Gtk.Entry.set_progress_pulse_step(), or Gtk.Entry.progress_pulse().
Additionally, an Entry can show icons at either side of the entry. These icons can be activatable by clicking, can be set
up as drag source and can have tooltips. To add an icon, use Gtk.Entry.set_icon_from_icon_name() or
one of the various other functions that set an icon from an icon name, a pixbuf, or icon theme. To set a tooltip on an
icon, use Gtk.Entry.set_icon_tooltip_text() or the corresponding function for markup.
7.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GObject
31
4
5
class EntryWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="Entry Demo")
self.set_size_request(200, 100)
7
8
9
10
self.timeout_id = None
11
12
13
14
15
self.entry = Gtk.Entry()
self.entry.set_text("Hello World")
vbox.pack_start(self.entry, True, True, 0)
16
17
18
19
hbox = Gtk.Box(spacing=6)
vbox.pack_start(hbox, True, True, 0)
20
21
22
self.check_editable = Gtk.CheckButton("Editable")
self.check_editable.connect("toggled", self.on_editable_toggled)
self.check_editable.set_active(True)
hbox.pack_start(self.check_editable, True, True, 0)
23
24
25
26
27
self.check_visible = Gtk.CheckButton("Visible")
self.check_visible.connect("toggled", self.on_visible_toggled)
self.check_visible.set_active(True)
hbox.pack_start(self.check_visible, True, True, 0)
28
29
30
31
32
self.pulse = Gtk.CheckButton("Pulse")
self.pulse.connect("toggled", self.on_pulse_toggled)
self.pulse.set_active(False)
hbox.pack_start(self.pulse, True, True, 0)
33
34
35
36
37
self.icon = Gtk.CheckButton("Icon")
self.icon.connect("toggled", self.on_icon_toggled)
self.icon.set_active(False)
hbox.pack_start(self.icon, True, True, 0)
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
32
Chapter 7. Entry
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
win = EntryWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
7.1. Example
33
34
Chapter 7. Entry
CHAPTER 8
Button Widgets
8.1 Button
The Button widget is another commonly used widget. It is generally used to attach a function that is called when the
button is pressed.
The Gtk.Button widget can hold any valid child widget. That is it can hold most any other standard Gtk.Widget.
The most commonly used child is the Gtk.Label.
Usually, you want to connect to the buttons clicked signal which is emitted when the button has been pressed and
released.
8.1.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class ButtonWindow(Gtk.Window):
6
7
8
9
def __init__(self):
Gtk.Window.__init__(self, title="Button Demo")
self.set_border_width(10)
10
11
12
hbox = Gtk.Box(spacing=6)
self.add(hbox)
13
14
15
16
17
35
button = Gtk.Button.new_with_mnemonic("_Open")
button.connect("clicked", self.on_open_clicked)
hbox.pack_start(button, True, True, 0)
18
19
20
21
button = Gtk.Button.new_with_mnemonic("_Close")
button.connect("clicked", self.on_close_clicked)
hbox.pack_start(button, True, True, 0)
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
win = ButtonWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
8.2 ToggleButton
A Gtk.ToggleButton is very similar to a normal Gtk.Button, but when clicked they remain activated, or
pressed, until clicked again. When the state of the button is changed, the toggled signal is emitted.
To retrieve the state of the Gtk.ToggleButton, you can use the Gtk.ToggleButton.get_active()
method. This returns True if the button is down. You can also set the toggle buttons state, with
Gtk.ToggleButton.set_active(). Note that, if you do this, and the state actually changes, it causes the
toggled signal to be emitted.
8.2.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class ToggleButtonWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="ToggleButton Demo")
self.set_border_width(10)
7
8
9
36
10
11
12
hbox = Gtk.Box(spacing=6)
self.add(hbox)
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
win = ToggleButtonWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
8.3 CheckButton
Gtk.CheckButton inherits from Gtk.ToggleButton. The only real difference between the two is
Gtk.CheckButtons appearance. A Gtk.CheckButton places a discrete Gtk.ToggleButton next
to a widget, (usually a Gtk.Label). The toggled signal, Gtk.ToggleButton.set_active() and
Gtk.ToggleButton.get_active() are inherited.
8.4 RadioButton
Like checkboxes, radio buttons also inherit from Gtk.ToggleButton, but these work in groups, and only one
Gtk.RadioButton in a group can be selected at any one time. Therefore, a Gtk.RadioButton is one way of
giving the user a choice from many options.
Radio buttons can be created with one of the static methods Gtk.RadioButton.new_from_widget(),
Gtk.RadioButton.new_with_label_from_widget() or Gtk.RadioButton.new_with_mnemonic_from_widget
The first radio button in a group will be created passing None as the group argument. In subsequent calls, the group
you wish to add this button to should be passed as an argument.
When first run, the first radio button in the group will be active.
Gtk.ToggleButton.set_active() with True as first argument.
Changing a Gtk.RadioButtons widget
Gtk.RadioButton.join_group().
8.3. CheckButton
group
after
its
creation
be
achieved
by
calling
37
8.4.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class RadioButtonWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="RadioButton Demo")
self.set_border_width(10)
7
8
9
10
hbox = Gtk.Box(spacing=6)
self.add(hbox)
11
12
13
14
15
16
17
button2 = Gtk.RadioButton.new_from_widget(button1)
button2.set_label("Button 2")
button2.connect("toggled", self.on_button_toggled, "2")
hbox.pack_start(button2, False, False, 0)
18
19
20
21
22
button3 = Gtk.RadioButton.new_with_mnemonic_from_widget(button1,
"B_utton 3")
button3.connect("toggled", self.on_button_toggled, "3")
hbox.pack_start(button3, False, False, 0)
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
win = RadioButtonWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
8.5 LinkButton
A Gtk.LinkButton is a Gtk.Button with a hyperlink, similar to the one used by web browsers, which triggers
an action when clicked. It is useful to show quick links to resources.
38
The URI bound to a Gtk.LinkButton can be set specifically using Gtk.LinkButton.set_uri(), and retrieved using Gtk.LinkButton.get_uri().
8.5.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class LinkButtonWindow(Gtk.Window):
6
7
8
9
def __init__(self):
Gtk.Window.__init__(self, title="LinkButton Demo")
self.set_border_width(10)
10
11
12
13
14
15
16
17
win = LinkButtonWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
8.6 SpinButton
A Gtk.SpinButton is an ideal way to allow the user to set the value of some attribute. Rather than having to
directly type a number into a Gtk.Entry, Gtk.SpinButton allows the user to click on one of two arrows to
increment or decrement the displayed value. A value can still be typed in, with the bonus that it can be checked to
ensure it is in a given range. The main properties of a Gtk.SpinButton are set through Gtk.Adjustment.
To change the value that Gtk.SpinButton is showing, use Gtk.SpinButton.set_value(). The value
entered can either be an integer or float, depending on your requirements, use Gtk.SpinButton.get_value()
or Gtk.SpinButton.get_value_as_int(), respectively.
When you allow the displaying of float values in the spin button, you may wish to adjust the number of decimal spaces
displayed by calling Gtk.SpinButton.set_digits().
By default, Gtk.SpinButton accepts textual data. If you wish to limit this to numerical values only, call
Gtk.SpinButton.set_numeric() with True as argument.
We can also adjust the update policy of Gtk.SpinButton. There are two options here; by default the spin button
updates the value even if the data entered is invalid. Alternatively, we can set the policy to only update when the value
entered is valid by calling Gtk.SpinButton.set_update_policy().
8.6. SpinButton
39
8.6.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class SpinButtonWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="SpinButton Demo")
self.set_border_width(10)
7
8
9
10
hbox = Gtk.Box(spacing=6)
self.add(hbox)
11
12
13
14
15
16
17
18
check_numeric = Gtk.CheckButton("Numeric")
check_numeric.connect("toggled", self.on_numeric_toggled)
hbox.pack_start(check_numeric, False, False, 0)
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
win = SpinButtonWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
40
8.7 Switch
A Gtk.Switch is a widget that has two states: on or off. The user can control which state should be active by
clicking the empty area, or by dragging the handle.
You shouldnt use the activate signal on the Gtk.Switch which is an action signal and emitting it causes the switch
to animate. Applications should never connect to this signal, but use the notify::active signal, see the example here
below.
8.7.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class SwitcherWindow(Gtk.Window):
6
7
8
9
def __init__(self):
Gtk.Window.__init__(self, title="Switch Demo")
self.set_border_width(10)
10
11
12
hbox = Gtk.Box(spacing=6)
self.add(hbox)
13
14
15
16
17
switch = Gtk.Switch()
switch.connect("notify::active", self.on_switch_activated)
switch.set_active(False)
hbox.pack_start(switch, True, True, 0)
18
19
20
21
22
switch = Gtk.Switch()
switch.connect("notify::active", self.on_switch_activated)
switch.set_active(True)
hbox.pack_start(switch, True, True, 0)
23
24
25
26
27
28
29
30
31
32
33
34
win = SwitcherWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
35
8.7. Switch
41
42
CHAPTER 9
ProgressBar
The Gtk.ProgressBar is typically used to display the progress of a long running operation. It provides a visual
clue that processing is underway. The Gtk.ProgressBar can be used in two different modes: percentage mode
and activity mode.
When an application can determine how much work needs to take place (e.g. read a fixed number of bytes from a file)
and can monitor its progress, it can use the Gtk.ProgressBar in percentage mode and the user sees a growing
bar indicating the percentage of the work that has been completed. In this mode, the application is required to call
Gtk.ProgressBar.set_fraction() periodically to update the progress bar, passing a float between 0 and 1
to provide the new percentage value.
When an application has no accurate way of knowing the amount of work to do, it can use activity mode, which shows
activity by a block moving back and forth within the progress area. In this mode, the application is required to call
Gtk.ProgressBar.pulse() periodically to update the progress bar. You can also choose the step size, with the
Gtk.ProgressBar.set_pulse_step() method.
By default, Gtk.ProgressBar is horizontal and left-to-right, but you can change it to a vertical progress bar by
using the Gtk.ProgressBar.set_orientation() method. Changing the direction the progress bar grows
can be done using Gtk.ProgressBar.set_inverted(). Gtk.ProgressBar can also contain text which
can be set by calling Gtk.ProgressBar.set_text() and Gtk.ProgressBar.set_show_text().
9.1 Example
1
2
import gi
gi.require_version('Gtk', '3.0')
43
4
5
class ProgressBarWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="ProgressBar Demo")
self.set_border_width(10)
7
8
9
10
11
12
13
self.progressbar = Gtk.ProgressBar()
vbox.pack_start(self.progressbar, True, True, 0)
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
44
Chapter 9. ProgressBar
if new_value > 1:
new_value = 0
61
62
63
self.progressbar.set_fraction(new_value)
64
65
66
67
68
69
70
71
72
73
win = ProgressBarWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
9.1. Example
45
46
Chapter 9. ProgressBar
CHAPTER 10
Spinner
10.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class SpinnerAnimation(Gtk.Window):
6
7
def __init__(self):
8
9
10
11
Gtk.Window.__init__(self, title="Spinner")
self.set_border_width(3)
self.connect("delete-event", Gtk.main_quit)
12
13
14
15
16
17
self.spinner = Gtk.Spinner()
18
19
20
47
self.table.attach(self.spinner, 0, 2, 2, 3)
21
22
self.add(self.table)
self.show_all()
23
24
25
26
27
if button.get_active():
self.spinner.start()
self.button.set_label("Stop Spinning")
28
29
30
31
else:
self.spinner.stop()
self.button.set_label("Start Spinning")
32
33
34
35
36
37
myspinner = SpinnerAnimation()
38
39
Gtk.main()
40
48
CHAPTER 11
A Gtk.TreeView and its associated widgets are an extremely powerful way of displaying data. They are used in
conjunction with a Gtk.ListStore or Gtk.TreeStore and provide a way of displaying and manipulating data
in many ways, including:
Automatically updates when data added, removed or edited
Drag and drop support
Sorting of data
Support embedding widgets such as check boxes, progress bars, etc.
Reorderable and resizable columns
Filtering of data
With the power and flexibility of a Gtk.TreeView comes complexity. It is often difficult for beginner developers to
be able to utilize it correctly due to the number of methods which are required.
This creates a list store with three columns, two string columns, and a float column.
Adding data to the model is done using Gtk.ListStore.append() or Gtk.TreeStore.append(), depending upon which sort of model was created.
treeiter = store.append(["The Art of Computer Programming",
"Donald E. Knuth", 25.46])
49
Both methods return a Gtk.TreeIter instance, which points to the location of the newly inserted row. You can
retrieve a Gtk.TreeIter by calling Gtk.TreeModel.get_iter().
Once data has been inserted, you can retrieve or modify data using the tree iter and column index.
print(store[treeiter][2]) # Prints value of third column
store[treeiter][2] = 42.15
As with Pythons built-in list object you can use len() to get the number of rows and use slices to retrieve or set
values.
# Print number of rows
print(len(store))
# Print all but first column
print(store[treeiter][1:])
# Print last column
print(store[treeiter][-1])
# Set first two columns
store[treeiter][:2] = ["Donald Ervin Knuth", 41.99]
Keep in mind, that if you use Gtk.TreeStore, the above code will only iterate over the rows of the top level, but
not the children of the nodes. To iterate over all rows and its children, use the print_tree_store function.
def print_tree_store(store):
rootiter = store.get_iter_first()
print_rows(store, rootiter, "")
def print_rows(store, treeiter, indent):
while treeiter != None:
print(indent + str(store[treeiter][:]))
if store.iter_has_child(treeiter):
childiter = store.iter_children(treeiter)
print_rows(store, childiter, indent + "\t")
treeiter = store.iter_next(treeiter)
Apart from accessing values stored in a Gtk.TreeModel with the list-like method mentioned above, it is also
possible to either use Gtk.TreeIter or Gtk.TreePath instances. Both reference a particular row in a tree
model. One can convert a path to an iterator by calling Gtk.TreeModel.get_iter(). As Gtk.ListStore
contains only one level, i.e. nodes do not have any child nodes, a path is essentially the index of the row you want to
access.
# Get path pointing to 6th row in list store
path = Gtk.TreePath(5)
treeiter = liststore.get_iter(path)
# Get value at 2nd column
value = liststore.get_value(treeiter, 1)
In the case of Gtk.TreeStore, a path is a list of indexes or a string. The string form is a list of numbers separated
by a colon. Each number refers to the offset at that level. Thus, the path 0 refers to the root node and the path 2:4
refers to the fifth child of the third node.
# Get path pointing to 5th child of 3rd row in tree store
path = Gtk.TreePath([2, 4])
treeiter = treestore.get_iter(path)
50
Instances of Gtk.TreePath can be accessed like lists, i.e. len(treepath) returns the depth of the item
treepath is pointing to, and treepath[i] returns the childs index on the i-th level.
Once the Gtk.TreeView widget has a model, it will need to know how to display the model. It does this with
columns and cell renderers.
Cell renderers are used to draw the data in the tree model in a way. There are a number of cell renderers that come with GTK+, for instance Gtk.CellRendererText, Gtk.CellRendererPixbuf and
Gtk.CellRendererToggle. In addition, it is relatively easy to write a custom renderer yourself.
A Gtk.TreeViewColumn is the object that Gtk.TreeView uses to organize the vertical columns in the tree view.
It needs to know the name of the column to label for the user, what type of cell renderer to use, and which piece of
data to retrieve from the model for a given row.
renderer = Gtk.CellRendererText()
column = Gtk.TreeViewColumn("Title", renderer, text=0)
tree.append_column(column)
To render more than one model column in a view column, you need to create a Gtk.TreeViewColumn instance
and use Gtk.TreeViewColumn.pack_start() to add the model columns to it.
column = Gtk.TreeViewColumn("Title and Author")
title = Gtk.CellRendererText()
author = Gtk.CellRendererText()
column.pack_start(title, True)
column.pack_start(author, True)
column.add_attribute(title, "text", 0)
column.add_attribute(author, "text", 1)
tree.append_column(column)
51
def on_tree_selection_changed(selection):
model, treeiter = selection.get_selected()
if treeiter != None:
print("You selected", model[treeiter][0])
11.4 Sorting
Sorting is an important feature for tree views and is supported by the standard tree models (Gtk.TreeStore and
Gtk.ListStore), which implement the Gtk.TreeSortable interface.
The next step is to enable sorting. Note that the column_id (0 in the example) refers to the column of the model and
not to the TreeViews column.
column.set_sort_column_id(0)
52
david
David
First of all a comparison function is needed. This function gets two rows and has to return a negative integer if the first
one should come before the second one, zero if they are equal and a positive integer if the second one should come
before the second one.
def compare(model, row1, row2, user_data):
sort_column, _ = model.get_sort_column_id()
value1 = model.get_value(row1, sort_column)
value2 = model.get_value(row2, sort_column)
if value1 < value2:
return -1
elif value1 == value2:
return 0
else:
return 1
11.5 Filtering
Unlike sorting, filtering is not handled by the two models we previously saw, but by the Gtk.TreeModelFilter
class. This class, like Gtk.TreeStore and Gtk.ListStore, is a Gtk.TreeModel. It acts as a layer between
the real model (a Gtk.TreeStore or a Gtk.ListStore), hiding some elements to the view. In practice, it
supplies the Gtk.TreeView with a subset of the underlying model. Instances of Gtk.TreeModelFilter can
be stacked one onto another, to use multiple filters on the same model (in the same way youd use AND clauses in a
SQL request). They can also be chained with Gtk.TreeModelSort instances.
You can create a new instance of a Gtk.TreeModelFilter and give it a model to filter, but the easiest way is to
spawn it directly from the filtered model, using the Gtk.TreeModel.filter_new() method.
filter = model.filter_new()
In the same way the sorting function works, the Gtk.TreeModelFilter needs a visibility function, which,
given a row from the underlying model, will return a boolean indicating if this row should be filtered out or not. Its
set by Gtk.TreeSortable.set_visible_func():
filter.set_visible_func(filter_func, data=None)
Lets look at a full example which uses the whole Gtk.ListStore - Gtk.TreeModelFilter Gtk.TreeModelFilter - Gtk.TreeView stack.
11.5. Filtering
53
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
6
7
8
9
10
11
12
13
14
15
16
#list of tuples for each software, containing the software name, initial release, and main programmin
software_list = [("Firefox", 2002, "C++"),
("Eclipse", 2004, "Java" ),
("Pitivi", 2004, "Python"),
("Netbeans", 1996, "Java"),
("Chrome", 2008, "C++"),
("Filezilla", 2001, "C++"),
("Bazaar", 2005, "Python"),
("Git", 2005, "C"),
("Linux Kernel", 1991, "C"),
("GCC", 1987, "C"),
("Frostwire", 2004, "Java")]
17
18
class TreeViewFilterWindow(Gtk.Window):
19
def __init__(self):
Gtk.Window.__init__(self, title="Treeview Filter Demo")
self.set_border_width(10)
20
21
22
23
24
25
26
27
28
29
30
31
32
54
33
34
self.software_liststore.append(list(software_ref))
self.current_filter_language = None
35
36
37
38
39
40
41
42
43
44
45
46
#creating the treeview, making it use the filter as a model, and adding the columns
self.treeview = Gtk.TreeView.new_with_model(self.language_filter)
for i, column_title in enumerate(["Software", "Release Year", "Programming Language"]):
renderer = Gtk.CellRendererText()
column = Gtk.TreeViewColumn(column_title, renderer, text=i)
self.treeview.append_column(column)
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
#setting up the layout, putting the treeview in a scrollwindow, and the buttons in a row
self.scrollable_treelist = Gtk.ScrolledWindow()
self.scrollable_treelist.set_vexpand(True)
self.grid.attach(self.scrollable_treelist, 0, 0, 8, 10)
self.grid.attach_next_to(self.buttons[0], self.scrollable_treelist, Gtk.PositionType.BOTTOM,
for i, button in enumerate(self.buttons[1:]):
self.grid.attach_next_to(button, self.buttons[i], Gtk.PositionType.RIGHT, 1, 1)
self.scrollable_treelist.add(self.treeview)
63
64
self.show_all()
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
win = TreeViewFilterWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
11.5. Filtering
55
56
CHAPTER 12
CellRenderers
Gtk.CellRenderer widgets are used to display information within widgets such as the Gtk.TreeView or
Gtk.ComboBox. They work closely with the associated widgets and are very powerful, with lots of configuration options for displaying a large amount of data in different ways. There are seven Gtk.CellRenderer widgets
which can be used for different purposes:
Gtk.CellRendererText
Gtk.CellRendererToggle
Gtk.CellRendererPixbuf
Gtk.CellRendererCombo
Gtk.CellRendererProgress
Gtk.CellRendererSpinner
Gtk.CellRendererSpin
Gtk.CellRendererAccel
12.1 CellRendererText
A Gtk.CellRendererText renders a given text in its cell, using the font, color and style information provided
by its properties. The text will be ellipsized if it is too long and the ellipsize property allows it.
By default, text in Gtk.CellRendererText widgets is not editable. This can be changed by setting the value of
the editable property to True:
cell.set_property("editable", True)
You can then connect to the edited signal and update your Gtk.TreeModel accordingly.
57
12.1.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class CellRendererTextWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererText Example")
7
8
9
self.set_default_size(200, 200)
10
11
12
13
14
15
16
treeview = Gtk.TreeView(model=self.liststore)
17
18
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Text", renderer_text, text=0)
treeview.append_column(column_text)
19
20
21
22
renderer_editabletext = Gtk.CellRendererText()
renderer_editabletext.set_property("editable", True)
23
24
25
26
27
28
29
renderer_editabletext.connect("edited", self.text_edited)
30
31
self.add(treeview)
32
58
33
34
35
36
37
38
39
40
win = CellRendererTextWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
12.2 CellRendererToggle
Gtk.CellRendererToggle renders a toggle button in a cell. The button is drawn as a radio- or checkbutton,
depending on the radio property. When activated, it emits the toggled signal.
As a Gtk.CellRendererToggle can have two states, active and not active, you most likely want to bind the
active property on the cell renderer to a boolean value in the model, thus causing the check button to reflect the state
of the model.
12.2.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class CellRendererToggleWindow(Gtk.Window):
6
7
8
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererToggle Example")
9
10
self.set_default_size(200, 200)
11
12.2. CellRendererToggle
59
12
13
14
15
16
treeview = Gtk.TreeView(model=self.liststore)
17
18
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Text", renderer_text, text=0)
treeview.append_column(column_text)
19
20
21
22
renderer_toggle = Gtk.CellRendererToggle()
renderer_toggle.connect("toggled", self.on_cell_toggled)
23
24
25
26
27
28
renderer_radio = Gtk.CellRendererToggle()
renderer_radio.set_radio(True)
renderer_radio.connect("toggled", self.on_cell_radio_toggled)
29
30
31
32
33
34
35
self.add(treeview)
36
37
38
39
40
41
42
43
44
45
46
47
48
49
win = CellRendererToggleWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
12.3 CellRendererPixbuf
A Gtk.CellRendererPixbuf can be used to render an image in a cell. It allows to render either a given
Gdk.Pixbuf (set via the pixbuf property) or a named icon (set via the icon-name property).
60
12.3.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class CellRendererPixbufWindow(Gtk.Window):
6
7
8
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererPixbuf Example")
9
10
self.set_default_size(200, 200)
11
12
13
14
15
16
17
treeview = Gtk.TreeView(model=self.liststore)
18
19
20
21
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Text", renderer_text, text=0)
treeview.append_column(column_text)
22
23
renderer_pixbuf = Gtk.CellRendererPixbuf()
24
25
26
27
28
self.add(treeview)
29
30
31
32
win = CellRendererPixbufWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
12.3. CellRendererPixbuf
61
33
Gtk.main()
12.4 CellRendererCombo
Gtk.CellRendererCombo renders text in a cell like Gtk.CellRendererText from which it is derived. But
while the latter offers a simple entry to edit the text, Gtk.CellRendererCombo offers a Gtk.ComboBox widget
to edit the text. The values to display in the combo box are taken from the Gtk.TreeModel specified in the model
property.
The combo cell renderer takes care of adding a text cell renderer to the combo box and sets it to display the column
specified by its text-column property.
A Gtk.CellRendererCombo can operate in two modes. It can be used with and without an associated
Gtk.Entry widget, depending on the value of the has-entry property.
12.4.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class CellRendererComboWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererCombo Example")
7
8
9
self.set_default_size(200, 200)
10
11
liststore_manufacturers = Gtk.ListStore(str)
manufacturers = ["Sony", "LG",
"Panasonic", "Toshiba", "Nokia", "Samsung"]
12
13
14
62
15
16
17
18
19
20
21
22
23
treeview = Gtk.TreeView(model=self.liststore_hardware)
24
25
26
27
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Text", renderer_text, text=0)
treeview.append_column(column_text)
28
29
30
31
32
33
34
renderer_combo = Gtk.CellRendererCombo()
renderer_combo.set_property("editable", True)
renderer_combo.set_property("model", liststore_manufacturers)
renderer_combo.set_property("text-column", 0)
renderer_combo.set_property("has-entry", False)
renderer_combo.connect("edited", self.on_combo_changed)
35
36
37
38
39
self.add(treeview)
40
41
42
43
44
45
46
47
win = CellRendererComboWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
12.5 CellRendererProgress
Gtk.CellRendererProgress renders a numeric value as a progress bar in a cell. Additionally, it can display a
text on top of the progress bar.
The percentage value of the progress bar can be modified by changing the value property. Similar to
Gtk.ProgressBar, you can enable the activity mode by incrementing the pulse property instead of the value
property.
12.5. CellRendererProgress
63
12.5.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GObject
4
5
class CellRendererProgressWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererProgress Example")
7
8
9
self.set_default_size(200, 200)
10
11
12
13
14
15
16
treeview = Gtk.TreeView(model=self.liststore)
17
18
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Text", renderer_text, text=0)
treeview.append_column(column_text)
19
20
21
22
renderer_progress = Gtk.CellRendererProgress()
column_progress = Gtk.TreeViewColumn("Progress", renderer_progress,
value=1, inverted=2)
treeview.append_column(column_progress)
23
24
25
26
27
renderer_toggle = Gtk.CellRendererToggle()
renderer_toggle.connect("toggled", self.on_inverted_toggled)
column_toggle = Gtk.TreeViewColumn("Inverted", renderer_toggle,
active=2)
treeview.append_column(column_toggle)
28
29
30
31
32
64
33
34
self.add(treeview)
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
self.liststore[self.current_iter][1] = new_value
return True
51
52
53
54
55
def reset_model(self):
for row in self.liststore:
row[1] = 0
self.current_iter = self.liststore.get_iter_first()
56
57
58
59
60
win = CellRendererProgressWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
12.6 CellRendererSpin
Gtk.CellRendererSpin renders text in a cell like Gtk.CellRendererText from which it is derived. But
while the latter offers a simple entry to edit the text, Gtk.CellRendererSpin offers a Gtk.SpinButton
widget. Of course, that means that the text has to be parseable as a floating point number.
The range of the spinbutton is taken from the adjustment property of the cell renderer, which can be set explicitly
or mapped to a column in the tree model, like all properties of cell renders. Gtk.CellRendererSpin also has
properties for the climb rate and the number of digits to display.
12.6. CellRendererSpin
65
12.6.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class CellRendererSpinWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="CellRendererSpin Example")
7
8
9
self.set_default_size(200, 200)
10
11
12
13
14
15
16
treeview = Gtk.TreeView(model=self.liststore)
17
18
renderer_text = Gtk.CellRendererText()
column_text = Gtk.TreeViewColumn("Fruit", renderer_text, text=0)
treeview.append_column(column_text)
19
20
21
22
renderer_spin = Gtk.CellRendererSpin()
renderer_spin.connect("edited", self.on_amount_edited)
renderer_spin.set_property("editable", True)
23
24
25
26
27
28
29
30
31
32
66
33
self.add(treeview)
34
35
36
37
38
39
40
41
win = CellRendererSpinWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
12.6. CellRendererSpin
67
68
CHAPTER 13
ComboBox
A Gtk.ComboBox allows for the selection of an item from a dropdown menu. They are preferable to having many
radio buttons on screen as they take up less room. If appropriate, it can show extra information about each item, such
as text, a picture, a checkbox, or a progress bar.
Gtk.ComboBox is very similar to Gtk.TreeView, as both use the model-view pattern; the list of valid choices is
specified in the form of a tree model, and the display of the choices can be adapted to the data in the model by using
cell renderers. If the combo box contains a large number of items, it may be better to display them in a grid rather
than a list. This can be done by calling Gtk.ComboBox.set_wrap_width().
The Gtk.ComboBox widget usually restricts the user to the available choices, but it can optionally have an Gtk.Entry, allowing the user to enter arbitrary text if none of the available choices
are suitable.
To do this, use one of the static methods Gtk.ComboBox.new_with_entry() or
Gtk.ComboBox.new_with_model_and_entry() to create an Gtk.ComboBox instance.
For a simple list of textual choices, the model-view API of Gtk.ComboBox can be a bit overwhelming. In this case,
Gtk.ComboBoxText offers a simple alternative. Both Gtk.ComboBox and Gtk.ComboBoxText can contain
an entry.
13.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
69
class ComboBoxWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="ComboBox Example")
7
8
9
self.set_border_width(10)
10
11
12
13
14
15
16
17
18
19
20
21
name_combo = Gtk.ComboBox.new_with_model_and_entry(name_store)
name_combo.connect("changed", self.on_name_combo_changed)
name_combo.set_entry_text_column(1)
vbox.pack_start(name_combo, False, False, 0)
22
23
24
25
26
country_store = Gtk.ListStore(str)
countries = ["Austria", "Brazil", "Belgium", "France", "Germany",
"Switzerland", "United Kingdom", "United States of America",
"Uruguay"]
for country in countries:
country_store.append([country])
27
28
29
30
31
32
33
country_combo = Gtk.ComboBox.new_with_model(country_store)
country_combo.connect("changed", self.on_country_combo_changed)
renderer_text = Gtk.CellRendererText()
country_combo.pack_start(renderer_text, True)
country_combo.add_attribute(renderer_text, "text", 0)
vbox.pack_start(country_combo, False, False, True)
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
self.add(vbox)
51
52
53
54
55
56
57
58
59
60
61
62
70
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
win = ComboBoxWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
13.1. Example
71
72
CHAPTER 14
IconView
A Gtk.IconView is a widget that displays a collection of icons in a grid view. It supports features such as drag and
drop, multiple selections and item reordering.
Similarly to Gtk.TreeView, Gtk.IconView uses a Gtk.ListStore for its model. Instead of using cell renderers, Gtk.IconView requires that one of the columns in its Gtk.ListStore contains GdkPixbuf.Pixbuf
objects.
Gtk.IconView supports numerous selection modes to allow for either selecting multiple icons at a time, restricting selections to just one item or disallowing selecting items completely. To specify a selection mode, the
Gtk.IconView.set_selection_mode() method is used with one of the Gtk.SelectionMode selection
modes.
73
14.1 Example
1
2
3
4
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
from gi.repository.GdkPixbuf import Pixbuf
5
6
7
8
class IconViewWindow(Gtk.Window):
9
10
11
12
def __init__(self):
Gtk.Window.__init__(self)
self.set_default_size(200, 200)
13
14
15
16
17
18
19
20
21
22
23
self.add(iconview)
24
25
26
win = IconViewWindow()
74
27
28
29
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
14.1. Example
75
76
CHAPTER 15
The Gtk.TextView widget can be used to display and edit large amounts of formatted text. Like the
Gtk.TreeView, it has a model/view design. In this case the Gtk.TextBuffer is the model which represents
the text being edited. This allows two or more Gtk.TextView widgets to share the same Gtk.TextBuffer, and
allows those text buffers to be displayed slightly differently. Or you could maintain several text buffers and choose to
display each one at different times in the same Gtk.TextView widget.
77
Many methods exist to retrieve a Gtk.TextIter. For instance, Gtk.TextBuffer.get_start_iter() returns an iterator pointing to the first position in the text buffer, whereas Gtk.TextBuffer.get_end_iter()
returns an iterator pointing past the last valid character. Retrieving the bounds of the selected text can be achieved by
calling Gtk.TextBuffer.get_selection_bounds().
To insert text at a specific position use Gtk.TextBuffer.insert().
Another useful method is
Gtk.TextBuffer.insert_at_cursor() which inserts text wherever the cursor may be currently positioned.
To remove portions of the text buffer use Gtk.TextBuffer.delete().
In addition, Gtk.TextIter can be used to locate textual matches in the buffer using
Gtk.TextIter.forward_search() and Gtk.TextIter.backward_search(). The start and end
iters are used as the starting point of the search and move forwards/backwards depending on requirements.
15.3 Tags
Text in a buffer can be marked with tags. A tag is an attribute that can be applied to some range of text. For example,
a tag might be called bold and make the text inside the tag bold. However, the tag concept is more general than
that; tags dont have to affect appearance. They can instead affect the behaviour of mouse and key presses, lock a
range of text so the user cant edit it, or countless other things. A tag is represented by a Gtk.TextTag object. One
Gtk.TextTag can be applied to any number of text ranges in any number of buffers.
Each tag is stored in a Gtk.TextTagTable. A tag table defines a set of tags that can be used together. Each buffer
has one tag table associated with it; only tags from that tag table can be used with the buffer. A single tag table can be
shared between multiple buffers, however.
To specify that some text in the buffer should have specific formatting, you must define a tag to hold that formatting information, and then apply that tag to the region of text using Gtk.TextBuffer.create_tag() and
Gtk.TextBuffer.apply_tag():
tag = textbuffer.create_tag("orange_bg", background="orange")
textbuffer.apply_tag(tag, start_iter, end_iter)
78
15.4 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Pango
4
5
class SearchDialog(Gtk.Dialog):
6
7
8
9
10
11
12
13
box = self.get_content_area()
14
15
16
17
18
self.entry = Gtk.Entry()
15.4. Example
79
box.add(self.entry)
19
20
self.show_all()
21
22
23
class TextViewWindow(Gtk.Window):
24
def __init__(self):
Gtk.Window.__init__(self, title="TextView Example")
25
26
27
self.set_default_size(-1, 350)
28
29
self.grid = Gtk.Grid()
self.add(self.grid)
30
31
32
self.create_textview()
self.create_toolbar()
self.create_buttons()
33
34
35
36
def create_toolbar(self):
toolbar = Gtk.Toolbar()
self.grid.attach(toolbar, 0, 0, 3, 1)
37
38
39
40
button_bold = Gtk.ToolButton()
button_bold.set_icon_name("format-text-bold-symbolic")
toolbar.insert(button_bold, 0)
41
42
43
44
button_italic = Gtk.ToolButton()
button_italic.set_icon_name("format-text-italic-symbolic")
toolbar.insert(button_italic, 1)
45
46
47
48
button_underline = Gtk.ToolButton()
button_underline.set_icon_name("format-text-underline-symbolic")
toolbar.insert(button_underline, 2)
49
50
51
52
53
54
55
56
57
58
toolbar.insert(Gtk.SeparatorToolItem(), 3)
59
60
radio_justifyleft = Gtk.RadioToolButton()
radio_justifyleft.set_icon_name("format-justify-left-symbolic")
toolbar.insert(radio_justifyleft, 4)
61
62
63
64
radio_justifycenter = Gtk.RadioToolButton.new_from_widget(radio_justifyleft)
radio_justifycenter.set_icon_name("format-justify-center-symbolic")
toolbar.insert(radio_justifycenter, 5)
65
66
67
68
radio_justifyright = Gtk.RadioToolButton.new_from_widget(radio_justifyleft)
radio_justifyright.set_icon_name("format-justify-right-symbolic")
toolbar.insert(radio_justifyright, 6)
69
70
71
72
radio_justifyfill = Gtk.RadioToolButton.new_from_widget(radio_justifyleft)
radio_justifyfill.set_icon_name("format-justify-fill-symbolic")
toolbar.insert(radio_justifyfill, 7)
73
74
75
76
80
77
78
79
80
81
82
83
84
radio_justifyleft.connect("toggled", self.on_justify_toggled,
Gtk.Justification.LEFT)
radio_justifycenter.connect("toggled", self.on_justify_toggled,
Gtk.Justification.CENTER)
radio_justifyright.connect("toggled", self.on_justify_toggled,
Gtk.Justification.RIGHT)
radio_justifyfill.connect("toggled", self.on_justify_toggled,
Gtk.Justification.FILL)
85
86
toolbar.insert(Gtk.SeparatorToolItem(), 8)
87
88
89
90
91
button_clear = Gtk.ToolButton()
button_clear.set_icon_name("edit-clear-symbolic")
button_clear.connect("clicked", self.on_clear_clicked)
toolbar.insert(button_clear, 9)
92
93
toolbar.insert(Gtk.SeparatorToolItem(), 10)
94
95
96
97
98
button_search = Gtk.ToolButton()
button_search.set_icon_name("system-search-symbolic")
button_search.connect("clicked", self.on_search_clicked)
toolbar.insert(button_search, 11)
99
100
101
102
103
104
def create_textview(self):
scrolledwindow = Gtk.ScrolledWindow()
scrolledwindow.set_hexpand(True)
scrolledwindow.set_vexpand(True)
self.grid.attach(scrolledwindow, 0, 1, 3, 1)
105
106
107
108
109
110
111
self.textview = Gtk.TextView()
self.textbuffer = self.textview.get_buffer()
self.textbuffer.set_text("This is some text inside of a Gtk.TextView. "
+ "Select text and click one of the buttons 'bold', 'italic', "
+ "or 'underline' to modify the text accordingly.")
scrolledwindow.add(self.textview)
112
113
114
115
116
117
118
119
120
self.tag_bold = self.textbuffer.create_tag("bold",
weight=Pango.Weight.BOLD)
self.tag_italic = self.textbuffer.create_tag("italic",
style=Pango.Style.ITALIC)
self.tag_underline = self.textbuffer.create_tag("underline",
underline=Pango.Underline.SINGLE)
self.tag_found = self.textbuffer.create_tag("found",
background="yellow")
121
122
123
124
125
126
def create_buttons(self):
check_editable = Gtk.CheckButton("Editable")
check_editable.set_active(True)
check_editable.connect("toggled", self.on_editable_toggled)
self.grid.attach(check_editable, 0, 2, 1, 1)
127
128
129
130
131
132
133
134
radio_wrapnone = Gtk.RadioButton.new_with_label_from_widget(None,
15.4. Example
81
"No Wrapping")
self.grid.attach(radio_wrapnone, 0, 3, 1, 1)
135
136
137
radio_wrapchar = Gtk.RadioButton.new_with_label_from_widget(
radio_wrapnone, "Character Wrapping")
self.grid.attach_next_to(radio_wrapchar, radio_wrapnone,
Gtk.PositionType.RIGHT, 1, 1)
138
139
140
141
142
radio_wrapword = Gtk.RadioButton.new_with_label_from_widget(
radio_wrapnone, "Word Wrapping")
self.grid.attach_next_to(radio_wrapword, radio_wrapchar,
Gtk.PositionType.RIGHT, 1, 1)
143
144
145
146
147
radio_wrapnone.connect("toggled", self.on_wrap_toggled,
Gtk.WrapMode.NONE)
radio_wrapchar.connect("toggled", self.on_wrap_toggled,
Gtk.WrapMode.CHAR)
radio_wrapword.connect("toggled", self.on_wrap_toggled,
Gtk.WrapMode.WORD)
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
self.search_and_mark(dialog.entry.get_text(), start)
187
188
dialog.destroy()
189
190
191
192
82
193
194
195
196
197
198
if match != None:
match_start, match_end = match
self.textbuffer.apply_tag(self.tag_found, match_start, match_end)
self.search_and_mark(text, match_end)
199
200
201
202
203
win = TextViewWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
15.4. Example
83
84
CHAPTER 16
Dialogs
Dialog windows are very similar to standard windows, and are used to provide or retrieve information from the
user. They are often used to provide a preferences window, for example. The major difference a dialog has is some
prepacked widgets which layout the dialog automatically. From there, we can simply add labels, buttons, check buttons, etc. Another big difference is the handling of responses to control how the application should behave after the
dialog has been interacted with.
There are several derived Dialog classes which you might find useful. Gtk.MessageDialog is used for most
simple notifications. But at other times you might need to derive your own dialog class to provide more complex
functionality.
85
16.1.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class DialogExample(Gtk.Dialog):
7
8
9
10
11
self.set_default_size(150, 100)
12
13
14
15
box = self.get_content_area()
box.add(label)
self.show_all()
16
17
18
19
20
class DialogWindow(Gtk.Window):
21
def __init__(self):
Gtk.Window.__init__(self, title="Dialog Example")
22
23
24
self.set_border_width(6)
25
26
27
28
29
self.add(button)
30
31
32
33
34
35
if response == Gtk.ResponseType.OK:
print("The OK button was clicked")
elif response == Gtk.ResponseType.CANCEL:
print("The Cancel button was clicked")
36
37
38
39
40
dialog.destroy()
41
42
43
win = DialogWindow()
86
44
45
46
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
16.2 MessageDialog
Gtk.MessageDialog is a convenience class, used to create simple, standard message dialogs, with a message, an
icon, and buttons for user response You can specify the type of message and the text in the Gtk.MessageDialog
constructor, as well as specifying standard buttons.
In some dialogs which require some further explanation of what has happened, a secondary text can be added. In
this case, the primary message entered when creating the message dialog is made bigger and set to bold text. The
secondary message can be set by calling Gtk.MessageDialog.format_secondary_text().
16.2.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class MessageDialogWindow(Gtk.Window):
6
7
8
def __init__(self):
Gtk.Window.__init__(self, title="MessageDialog Example")
9
10
11
box = Gtk.Box(spacing=6)
self.add(box)
12
13
14
15
button1 = Gtk.Button("Information")
button1.connect("clicked", self.on_info_clicked)
box.add(button1)
16
17
18
19
button2 = Gtk.Button("Error")
button2.connect("clicked", self.on_error_clicked)
box.add(button2)
20
21
22
23
button3 = Gtk.Button("Warning")
button3.connect("clicked", self.on_warn_clicked)
box.add(button3)
16.2. MessageDialog
87
24
button4 = Gtk.Button("Question")
button4.connect("clicked", self.on_question_clicked)
box.add(button4)
25
26
27
28
29
30
31
32
33
34
35
36
dialog.destroy()
37
38
39
40
41
42
43
44
45
46
dialog.destroy()
47
48
49
50
51
52
53
54
55
56
57
58
59
dialog.destroy()
60
61
62
63
64
65
66
67
68
69
70
71
72
dialog.destroy()
73
74
75
76
77
78
win = MessageDialogWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
88
16.3 FileChooserDialog
The Gtk.FileChooserDialog is suitable for use with File/Open or File/Save menu items. You can use all
of the Gtk.FileChooser methods on the file chooser dialog as well as those for Gtk.Dialog.
When creating a Gtk.FileChooserDialog you have to define the dialogs purpose:
To select a file for opening, as for a File/Open command, use Gtk.FileChooserAction.OPEN
To save a file for the first time, as for a File/Save command, use Gtk.FileChooserAction.SAVE, and
suggest a name such as Untitled with Gtk.FileChooser.set_current_name().
To
save
a
file
under
a
different
use
Gtk.FileChooserAction.SAVE,
Gtk.FileChooser.set_filename().
name,
and
as
set
for
a
File/Save
As
command,
the
existing
filename
with
16.3. FileChooserDialog
89
16.3.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class FileChooserWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title="FileChooser Example")
7
8
9
box = Gtk.Box(spacing=6)
self.add(box)
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
self.add_filters(dialog)
27
28
response = dialog.run()
if response == Gtk.ResponseType.OK:
print("Open clicked")
print("File selected: " + dialog.get_filename())
elif response == Gtk.ResponseType.CANCEL:
print("Cancel clicked")
29
30
31
32
33
34
90
35
36
dialog.destroy()
37
38
39
40
41
42
43
44
45
46
47
filter_py = Gtk.FileFilter()
filter_py.set_name("Python files")
filter_py.add_mime_type("text/x-python")
dialog.add_filter(filter_py)
48
49
50
51
52
filter_any = Gtk.FileFilter()
filter_any.set_name("Any files")
filter_any.add_pattern("*")
dialog.add_filter(filter_any)
53
54
55
56
57
58
59
60
61
62
63
64
65
66
response = dialog.run()
if response == Gtk.ResponseType.OK:
print("Select clicked")
print("Folder selected: " + dialog.get_filename())
elif response == Gtk.ResponseType.CANCEL:
print("Cancel clicked")
67
68
dialog.destroy()
69
70
71
72
73
win = FileChooserWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
16.3. FileChooserDialog
91
92
CHAPTER 17
Clipboard
Gtk.Clipboard provides a storage area for a variety of data, including text and images. Using a clipboard allows
this data to be shared between applications through actions such as copying, cutting, and pasting. These actions are
usually done in three ways: using keyboard shortcuts, using a Gtk.MenuItem, and connecting the functions to
Gtk.Button widgets.
There are multiple clipboard selections for different purposes. In most circumstances, the selection named
CLIPBOARD is used for everyday copying and pasting. PRIMARY is another common selection which stores text
selected by the user with the cursor.
17.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Gdk
4
5
class ClipboardWindow(Gtk.Window):
6
7
8
def __init__(self):
Gtk.Window.__init__(self, title="Clipboard Example")
9
10
table = Gtk.Table(3, 2)
11
12
13
14
self.clipboard = Gtk.Clipboard.get(Gdk.SELECTION_CLIPBOARD)
self.entry = Gtk.Entry()
self.image = Gtk.Image.new_from_icon_name("process-stop", Gtk.IconSize.MENU)
15
16
17
18
93
19
20
table.attach(self.entry, 0, 1, 0, 1)
table.attach(self.image, 0, 1, 1, 2)
table.attach(button_copy_text, 1, 2, 0, 1)
table.attach(button_paste_text, 2, 3, 0, 1)
table.attach(button_copy_image, 1, 2, 1, 2)
table.attach(button_paste_image, 2, 3, 1, 2)
21
22
23
24
25
26
27
button_copy_text.connect("clicked", self.copy_text)
button_paste_text.connect("clicked", self.paste_text)
button_copy_image.connect("clicked", self.copy_image)
button_paste_image.connect("clicked", self.paste_image)
28
29
30
31
32
self.add(table)
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
win = ClipboardWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
94
CHAPTER 18
Note: Versions of PyGObject < 3.0.3 contain a bug which does not allow drag and drop to function correctly.
Therefore a version of PyGObject >= 3.0.3 is required for the following examples to work.
Setting up drag and drop between widgets consists of selecting a drag source (the widget which the user starts the
drag from) with the Gtk.Widget.drag_source_set() method, selecting a drag destination (the widget which
the user drops onto) with the Gtk.Widget.drag_dest_set() method and then handling the relevant signals on
both widgets.
Instead of using Gtk.Widget.drag_source_set() and Gtk.Widget.drag_dest_set() some specialised widgets require the use of specific functions (such as Gtk.TreeView and Gtk.IconView).
A basic drag and drop only requires the source to connect to the drag-data-get signal and the destination to connect
to the drag-data-received signal. More complex things such as specific drop areas and custom drag icons will require
you to connect to additional signals and interact with the Gdk.DragContext object it supplies.
In order to transfer data between the source and destination, you must interact with the Gtk.SelectionData
variable supplied in the drag-data-get and drag-data-received signals using the Gtk.SelectionData get
and set methods.
95
When it is emitted
User starts a drag
When drag data is requested by the destination
When a drag with the action Gdk.DragAction.MOVE
is completed
When the drag is complete
Common Purpose
Set-up drag icon
Transfer drag data from source to
destination
Delete data from the source to complete
the move
Undo anything done in drag-begin
When it is emitted
Drag icon moves over a drop area
Icon is dropped onto a drag area
When drag data is received by the
destination
Common Purpose
Allow only certain areas to be dropped onto
Allow only certain areas to be dropped onto
Transfer drag data from source to
destination
18.4 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Gdk, GdkPixbuf
4
5
6
7
8
DRAG_ACTION = Gdk.DragAction.COPY
96
9
10
class DragDropWindow(Gtk.Window):
11
12
13
def __init__(self):
Gtk.Window.__init__(self, title="Drag and Drop Demo")
14
15
16
17
18
19
hbox = Gtk.Box(spacing=12)
vbox.pack_start(hbox, True, True, 0)
20
21
22
self.iconview = DragSourceIconView()
self.drop_area = DropArea()
23
24
25
26
27
28
button_box = Gtk.Box(spacing=6)
vbox.pack_start(button_box, True, False, 0)
29
30
31
32
33
image_button = Gtk.RadioButton.new_with_label_from_widget(None,
"Images")
image_button.connect("toggled", self.add_image_targets)
button_box.pack_start(image_button, True, False, 0)
34
35
36
37
38
text_button = Gtk.RadioButton.new_with_label_from_widget(image_button,
"Text")
text_button.connect("toggled", self.add_text_targets)
button_box.pack_start(text_button, True, False, 0)
39
40
self.add_image_targets()
41
42
43
44
45
46
47
self.drop_area.drag_dest_set_target_list(targets)
self.iconview.drag_source_set_target_list(targets)
48
49
50
51
52
53
54
self.drop_area.drag_dest_add_text_targets()
self.iconview.drag_source_add_text_targets()
55
56
class DragSourceIconView(Gtk.IconView):
57
58
59
60
61
def __init__(self):
Gtk.IconView.__init__(self)
self.set_text_column(COLUMN_TEXT)
self.set_pixbuf_column(COLUMN_PIXBUF)
62
63
64
65
66
18.4. Example
97
67
68
self.enable_model_drag_source(Gdk.ModifierType.BUTTON1_MASK, [],
DRAG_ACTION)
self.connect("drag-data-get", self.on_drag_data_get)
69
70
71
72
73
74
75
76
if info == TARGET_ENTRY_TEXT:
text = self.get_model().get_value(selected_iter, COLUMN_TEXT)
data.set_text(text, -1)
elif info == TARGET_ENTRY_PIXBUF:
pixbuf = self.get_model().get_value(selected_iter, COLUMN_PIXBUF)
data.set_pixbuf(pixbuf)
77
78
79
80
81
82
83
84
85
86
87
88
89
class DropArea(Gtk.Label):
90
def __init__(self):
Gtk.Label.__init__(self, "Drop something on me!")
self.drag_dest_set(Gtk.DestDefaults.ALL, [], DRAG_ACTION)
91
92
93
94
self.connect("drag-data-received", self.on_drag_data_received)
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
win = DragDropWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
98
CHAPTER 19
The Gtk.Builder class offers you the opportunity to design user interfaces without writing a single line of code.
This is possible through describing the interface by a XML file and then loading the XML description at runtime and
create the objects automatically, which the Builder class does for you. For the purpose of not needing to write the
XML manually the Glade application lets you create the user interface in a WYSIWYG (what you see is what you
get) manner
This method has several advantages:
Less code needs to be written.
UI changes can be seen more quickly, so UIs are able to improve.
Designers without programming skills can create and edit UIs.
The description of the user interface is independent from the programming language being used.
There is still code required for handling interface changes triggered by the user, but Gtk.Builder allows you to
focus on implementing that functionality.
99
The second line loads all objects defined in example.glade into the Builder object.
It is also possible to load only some of the objects. The following line would add only the objects (and their child
objects) given in the tuple.
# we don't really have two buttons here, this is just an example
builder.add_objects_from_file("example.glade", ("button1", "button2"))
These two methods exist also for loading from a string rather than a file. Their corresponding names are
Gtk.Builder.add_from_string() and Gtk.Builder.add_objects_from_string() and they simply take a XML string instead of a file name.
Every widget can be retrieved from the builder by the Gtk.Builder.get_object() method and the widgets
id. It is really that simple.
It is also possible to get a list of all objects with
builder.get_objects()
100
Now we have to define the handler functions in our code. The onDeleteWindow should simply result in a call to
Gtk.main_quit(). When the button is pressed we would like to print the string Hello World!, so we define the
handler as follows
def hello(button):
print("Hello World!")
Next, we have to connect the signals and the handler functions. The easiest way to do this is to define a dict with a
mapping from the names to the handlers and then pass it to the Gtk.Builder.connect_signals() method.
handlers = {
"onDeleteWindow": Gtk.main_quit,
"onButtonPressed": hello
}
builder.connect_signals(handlers)
An alternative approach is to create a class which has methods that are called like the signals. In our example the last
code snippet could be rewritten as:
1
2
3
4
5
class Handler:
def onDeleteWindow(self, *args):
Gtk.main_quit(*args)
6
7
8
9
10
builder = Gtk.Builder()
19.4 Example
The final code of the example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
6
7
class Handler:
def onDeleteWindow(self, *args):
Gtk.main_quit(*args)
8
9
10
11
12
13
14
builder = Gtk.Builder()
builder.add_from_file("builder_example.glade")
builder.connect_signals(Handler())
15
16
17
window = builder.get_object("window1")
window.show_all()
19.4. Example
101
18
19
Gtk.main()
102
CHAPTER 20
Objects
GObject is the fundamental type providing the common attributes and methods for all object types in GTK+, Pango
and other libraries based on GObject. The GObject.GObject class provides methods for object construction and
destruction, property access methods, and signal support.
This section will introduce some important aspects about the GObject implementation in Python.
20.2 Signals
Signals connect arbitrary application-specific events with any number of listeners. For example, in GTK+, every user
event (keystroke or mouse move) is received from the X server and generates a GTK+ event under the form of a signal
emission on a given object instance.
Each signal is registered in the type system together with the type on which it can be emitted: users of the type are
said to connect to the signal on a given type instance when they register a function to be invoked upon the signal
emission. Users can also emit the signal by themselves or stop the emission of the signal from within one of the
functions connected to the signal.
103
GObject.SIGNAL_RUN_FIRST indicates that this signal will invoke the object method handler
(do_my_signal() here) in the first emission stage. Alternatives are GObject.SIGNAL_RUN_LAST (the
method handler will be invoked in the third emission stage) and GObject.SIGNAL_RUN_CLEANUP (invoke the
method handler in the last emission stage).
The second part, None, indicates the return type of the signal, usually None.
(int,) indicates the signal arguments, here, the signal will only take one argument, whose type is int. This argument
type list must end with a comma.
Signals can be emitted using GObject.GObject.emit():
my_obj.emit("my_signal", 42) # emit the signal "my_signal", with the
# argument 42
20.3 Properties
One of GObjects nice features is its generic get/set mechanism for object properties. Each class inherited from
GObject.GObject can define new properties. Each property has a type which never changes (e.g. str, float, int...).
For instance, they are used for Gtk.Button where there is a label property which contains the text of the button.
properties,
Some properties also have functions dedicated to them, called getter and setter. For the property label of a button,
there are two functions to get and set them, Gtk.Button.get_label() and Gtk.Button.set_label().
104
property_float = GObject.Property(type=float)
def __init__(self):
GObject.GObject.__init__(self)
Properties can also be read-only, if you want some properties to be readable but not writable.
To
do so, you can add some flags to the property definition, to control read/write access.
Flags are
GObject.PARAM_READABLE (only read access for external code), GObject.PARAM_WRITABLE (only write
access), GObject.PARAM_READWRITE (public):
foo = GObject.Property(type=str, flags = GObject.PARAM_READABLE) # not writable
bar = GObject.Property(type=str, flags = GObject.PARAM_WRITABLE) # not readable
You can also define new read-only properties with a new method decorated with GObject.Property:
from gi.repository import GObject
class MyObject(GObject.GObject):
def __init__(self):
GObject.GObject.__init__(self)
@GObject.Property
def readonly(self):
return 'This is read-only.'
The API of GObject.Property is similar to the builtin property(). You can create property setter in a way
similar to Python property:
class AnotherObject(GObject.Object):
value = 0
@GObject.Property
def prop(self):
'Read only property.'
return 1
@GObject.Property(type=int)
def propInt(self):
'Read-write integer property.'
return self.value
@propInt.setter
def propInt(self, value):
self.value = value
There is also a way to define minimum and maximum values for numbers, using a more verbose form:
from gi.repository import GObject
class MyObject(GObject.GObject):
__gproperties__ = {
"int-prop": (int, # type
20.3. Properties
105
Note that you have to use the canonical property name when connecting to the notify signals, as explained in
GObject.Object.signals.notify(). For instance, for a Python property foo_bar_baz you would connect
to the signal notify::foo-bar-baz using
my_object = MyObject()
def on_notify_foo_bar_baz(obj, gparamstring):
print("foo_bar_baz changed")
my_object.connect("notify::foo-bar-baz", on_notify_foo_bar_baz)
106
20.4 API
class GObject.GObject
get_property(property_name)
Retrieves a property value.
set_property(property_name, value)
Set property property_name to value.
emit(signal_name, ...)
Emit signal signal_name. Signal arguments must follow, e.g. if your signal is of type (int,), it must be
emitted with:
self.emit(signal_name, 42)
freeze_notify()
This method freezes all the notify:: signals (which are emitted when any property is changed) until the
thaw_notify() method is called.
It recommended to use the with statement when calling freeze_notify(), that way it is ensured that
thaw_notify() is called implicitly at the end of the block:
with an_object.freeze_notify():
# Do your work here
...
thaw_notify()
Thaw all the notify:: signals which were thawed by freeze_notify().
It is recommended to not call thaw_notify() explicitly but use freeze_notify() together with
the with statement.
handler_block(handler_id)
Blocks a handler of an instance so it will not be called during any signal emissions unless
handler_unblock() is called for that handler_id. Thus blocking a signal handler means to temporarily deactivate it, a signal handler has to be unblocked exactly the same amount of times it has been
blocked before to become active again.
It is recommended to use handler_block() in conjunction with the with statement which will call
handler_unblock() implicitly at the end of the block:
with an_object.handler_block(handler_id):
# Do your work here
...
handler_unblock(handler_id)
Undoes the effect of handler_block(). A blocked handler is skipped during signal emissions and
will not be invoked until it has been unblocked exactly the amount of times it has been blocked before.
It is recommended to not call handler_unblock() explicitly but use handler_block() together
with the with statement.
__gsignals__
A dictionary where inherited class can define new signals.
Each element in the dictionary is a new signal. The key is the signal name. The value is a tuple, with the
form:
20.4. API
107
108
CHAPTER 21
Application
Gtk.Application encompeses many repetitive tasks that a modern application needs such as handling multiple instances, D-Bus activation, opening files, command line parsing, startup/shutdown, menu management, window
management, and more.
21.1 Actions
Gio.Action is a way to expose any single task your application or widget does by a name. These actions can be
disabled/enabled at runtime and they can either be activated or have a state changed (if they contain state).
The reason to use actions is to separate out the logic from the UI. For example this allows using a menubar on OSX
and a gear menu on GNOME both simply referencing the name of an action. The main implementation of this you
will be using is Gio.SimpleAction which will be showed off later.
Many classes such as Gio.MenuItem and Gtk.ModelButton support properties to set an action name.
These actions can be grouped together into a Gio.ActionGroup and when these groups are added to a widget with Gtk.Widget.insert_action_group() they will gain a prefix. Such as win when added to a
Gtk.ApplicationWindow. You will use the full action name when referencing it such as app.about but when
you create the action it will just be about until added to the application.
You can also very easily make keybindings for actions by setting the accel property in the Gio.Menu file or by using
Gtk.Application.add_accelerator().
21.2 Menus
Your menus should be defined in XML using Gio.Menu and would reference the previously mentioned actions
you defined. Gtk.Application allows you to set a menu either via Gtk.Application.set_app_menu()
or Gtk.Application.set_menubar(). If you make use of Gio.Resource this can automatically use the
correct menu based on platform, otherwise you can set them manually. A detailed example is shown below.
109
You
can
use
HANDLES_COMMAND_LINE
to
allow
custom
behavior
in
Gio.Application.do_command_line(). In combination with Gio.Application.add_main_option()
to add custom options.
Using HANDLES_OPEN will do the work of simply taking file arguments for you and let you handle it in
Gio.Application.do_open().
If your application is already open these will all be sent to the existing instance unless you use NON_UNIQUE to
allow multiple instances.
21.4 Example
import sys
2
3
4
5
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import GLib, Gio, Gtk
6
7
8
9
10
110
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
<menu id="app-menu">
<section>
<attribute name="label" translatable="yes">Change label</attribute>
<item>
<attribute name="action">win.change_label</attribute>
<attribute name="target">String 1</attribute>
<attribute name="label" translatable="yes">String 1</attribute>
</item>
<item>
<attribute name="action">win.change_label</attribute>
<attribute name="target">String 2</attribute>
<attribute name="label" translatable="yes">String 2</attribute>
</item>
<item>
<attribute name="action">win.change_label</attribute>
<attribute name="target">String 3</attribute>
<attribute name="label" translatable="yes">String 3</attribute>
</item>
</section>
<section>
<item>
<attribute name="action">win.maximize</attribute>
<attribute name="label" translatable="yes">Maximize</attribute>
</item>
</section>
<section>
<item>
<attribute name="action">app.about</attribute>
<attribute name="label" translatable="yes">_About</attribute>
</item>
<item>
<attribute name="action">app.quit</attribute>
<attribute name="label" translatable="yes">_Quit</attribute>
<attribute name="accel"><Primary>q</attribute>
</item>
</section>
</menu>
</interface>
"""
50
51
class AppWindow(Gtk.ApplicationWindow):
52
53
54
55
56
57
58
59
60
# This will be in the windows group and have the "win" prefix
max_action = Gio.SimpleAction.new_stateful("maximize", None,
GLib.Variant.new_boolean(False))
max_action.connect("change-state", self.on_maximize_toggle)
self.add_action(max_action)
61
62
63
64
65
66
67
68
21.4. Example
111
lbl_variant)
lbl_action.connect("change-state", self.on_change_label_state)
self.add_action(lbl_action)
69
70
71
72
self.label = Gtk.Label(label=lbl_variant.get_string(),
margin=30)
self.add(self.label)
self.label.show()
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
class Application(Gtk.Application):
90
91
92
93
94
95
96
97
98
99
def do_startup(self):
Gtk.Application.do_startup(self)
100
101
102
103
104
105
106
107
108
109
110
111
112
113
def do_activate(self):
# We only allow a single window and raise any existing ones
if not self.window:
# Windows are associated with the application
# when the last one is closed the application shuts down
self.window = AppWindow(application=self, title="Main Window")
114
115
116
117
118
119
120
self.window.present()
121
122
123
124
125
if options.contains("test"):
126
112
127
128
129
130
131
self.activate()
return 0
132
133
134
135
136
137
138
139
140
141
142
if __name__ == "__main__":
app = Application()
app.run(sys.argv)
113
114
CHAPTER 22
Menus
Note: Gtk.UIManager, Gtk.Action, and Gtk.ActionGroup have been deprecated since GTK+ version
3.10 and should not be used in newly-written code. Use the Application framework instead.
GTK+ comes with two different types of menus, Gtk.MenuBar and Gtk.Toolbar. Gtk.MenuBar is a standard
menu bar which contains one or more Gtk.MenuItem instances or one of its subclasses. Gtk.Toolbar widgets
are used for quick accessibility to commonly used functions of an application. Examples include creating a new
document, printing a page or undoing an operation. It contains one or more instances of Gtk.ToolItem or one of
its subclasses.
22.1 Actions
Although, there are specific APIs to create menus and toolbars, you should use Gtk.UIManager and create
Gtk.Action instances. Actions are organised into groups. A Gtk.ActionGroup is essentially a map from
names to Gtk.Action objects. All actions that would make sense to use in a particular context should be in a single
group. Multiple action groups may be used for a particular user interface. In fact, it is expected that most non-trivial
applications will make use of multiple groups. For example, in an application that can edit multiple documents, one
group holding global actions (e.g. quit, about, new), and one group per document holding actions that act on that
document (eg. save, cut/copy/paste, etc). Each windows menus would be constructed from a combination of two
action groups.
Different classes representing different types of actions exist:
Gtk.Action: An action which can be triggered by a menu or toolbar item
Gtk.ToggleAction: An action which can be toggled between two states
Gtk.RadioAction: An action of which only one in a group can be active
Gtk.RecentAction: An action of which represents a list of recently used files
Actions represent operations that the user can be perform, along with some information how it should be presented
in the interface, including its name (not for display), its label (for display), an accelerator, whether a label indicates a
tooltip as well as the callback that is called when the action gets activated.
You can create actions by either calling one of the constructors directly and adding them to a Gtk.ActionGroup
by calling Gtk.ActionGroup.add_action() or Gtk.ActionGroup.add_action_with_accel(), or
by calling one of the convenience functions:
Gtk.ActionGroup.add_actions(),
Gtk.ActionGroup.add_toggle_actions()
115
Gtk.ActionGroup.add_radio_actions().
Note that you must specify actions for sub menus as well as menu items.
22.2 UI Manager
Gtk.UIManager provides an easy way of creating menus and toolbars using an XML-like description.
First
of
all,
you
should
add
the
Gtk.ActionGroup
to
the
UI
Manager
with
Gtk.UIManager.insert_action_group(). At this point is also a good idea to tell the parent window to respond to the specified keyboard shortcuts, by using Gtk.UIManager.get_accel_group() and
Gtk.Window.add_accel_group().
Then, you can define the actual visible layout of the menus and toolbars, and add the UI layout. This ui string uses
an XML format, in which you should mention the names of the actions that you have already created. Remember that
these names are just the identifiers that we used when creating the actions. They are not the text that the user will see
in the menus and toolbars. We provided those human-readable names when we created the actions.
Finally, you retrieve the root widget with Gtk.UIManager.get_widget() and add the widget to a container
such as Gtk.Box.
22.3 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, Gdk
4
5
6
7
8
UI_INFO = """
<ui>
<menubar name='MenuBar'>
<menu action='FileMenu'>
116
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
<menu action='FileNew'>
<menuitem action='FileNewStandard' />
<menuitem action='FileNewFoo' />
<menuitem action='FileNewGoo' />
</menu>
<separator />
<menuitem action='FileQuit' />
</menu>
<menu action='EditMenu'>
<menuitem action='EditCopy' />
<menuitem action='EditPaste' />
<menuitem action='EditSomething' />
</menu>
<menu action='ChoicesMenu'>
<menuitem action='ChoiceOne'/>
<menuitem action='ChoiceTwo'/>
<separator />
<menuitem action='ChoiceThree'/>
</menu>
</menubar>
<toolbar name='ToolBar'>
<toolitem action='FileNewStandard' />
<toolitem action='FileQuit' />
</toolbar>
<popup name='PopupMenu'>
<menuitem action='EditCopy' />
<menuitem action='EditPaste' />
<menuitem action='EditSomething' />
</popup>
</ui>
"""
40
41
class MenuExampleWindow(Gtk.Window):
42
43
44
def __init__(self):
Gtk.Window.__init__(self, title="Menu Example")
45
46
self.set_default_size(200, 200)
47
48
action_group = Gtk.ActionGroup("my_actions")
49
50
51
52
self.add_file_menu_actions(action_group)
self.add_edit_menu_actions(action_group)
self.add_choices_menu_actions(action_group)
53
54
55
uimanager = self.create_ui_manager()
uimanager.insert_action_group(action_group)
56
57
menubar = uimanager.get_widget("/MenuBar")
58
59
60
box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
box.pack_start(menubar, False, False, 0)
61
62
63
toolbar = uimanager.get_widget("/ToolBar")
box.pack_start(toolbar, False, False, 0)
64
65
66
eventbox = Gtk.EventBox()
eventbox.connect("button-press-event", self.on_button_press_event)
22.3. Example
117
67
68
69
70
71
self.popup = uimanager.get_widget("/PopupMenu")
72
73
self.add(box)
74
75
76
77
78
79
80
81
82
83
84
85
86
87
action_group.add_actions([
("FileNewFoo", None, "New Foo", None, "Create new foo",
self.on_menu_file_new_generic),
("FileNewGoo", None, "_New Goo", None, "Create new goo",
self.on_menu_file_new_generic),
])
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
action_group.add_radio_actions([
("ChoiceOne", None, "One", None, None, 1),
("ChoiceTwo", None, "Two", None, None, 2)
], 1, self.on_menu_choices_changed)
114
115
116
117
118
119
120
121
122
def create_ui_manager(self):
uimanager = Gtk.UIManager()
123
124
118
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
window = MenuExampleWindow()
window.connect("delete-event", Gtk.main_quit)
window.show_all()
Gtk.main()
22.3. Example
119
120
CHAPTER 23
Table
Note: Gtk.Table has been deprecated since GTK+ version 3.4 and should not be used in newly-written code. Use
the Grid class instead.
Tables allows us to place widgets in a grid similar to Gtk.Grid.
The grids dimensions need to be specified in the Gtk.Table constructor. To place a widget into a box, use
Gtk.Table.attach().
Gtk.Table.set_row_spacing() and Gtk.Table.set_col_spacing() set the spacing between the
rows at the specified row or column. Note that for columns, the space goes to the right of the column, and for
rows, the space goes below the row.
You can also set a consistent spacing for all rows and/or columns with Gtk.Table.set_row_spacings() and
Gtk.Table.set_col_spacings(). Note that with these calls, the last row and last column do not get any
spacing.
Deprecated since version 3.4: It is reccomened that you use the Gtk.Grid for new code.
23.1 Example
1
2
3
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk
4
5
class TableWindow(Gtk.Window):
121
def __init__(self):
Gtk.Window.__init__(self, title="Table Example")
7
8
9
10
11
12
button1
button2
button3
button4
button5
button6
13
14
15
16
17
18
=
=
=
=
=
=
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
Gtk.Button(label="Button
1")
2")
3")
4")
5")
6")
19
table.attach(button1,
table.attach(button2,
table.attach(button3,
table.attach(button4,
table.attach(button5,
table.attach(button6,
20
21
22
23
24
25
0,
1,
0,
1,
1,
2,
1,
3,
1,
3,
2,
3,
0,
0,
1,
1,
2,
2,
1)
1)
3)
2)
3)
3)
26
27
28
29
30
win = TableWindow()
win.connect("delete-event", Gtk.main_quit)
win.show_all()
Gtk.main()
122
CHAPTER 24
search
123
124
Index
Symbols
__gproperties__ (GObject.GObject attribute), 108
__gsignals__ (GObject.GObject attribute), 107
E
emit() (GObject.GObject method), 107
F
freeze_notify() (GObject.GObject method), 107
G
get_property() (GObject.GObject method), 107
GObject.GObject (built-in class), 107
H
handler_block() (GObject.GObject method), 107
handler_unblock() (GObject.GObject method), 107
P
PARAM_READABLE (GObject attribute), 108
PARAM_READWRITE (GObject attribute), 108
PARAM_WRITABLE (GObject attribute), 108
S
set_property() (GObject.GObject method), 107
SIGNAL_RUN_CLEANUP (GObject attribute), 108
SIGNAL_RUN_FIRST (GObject attribute), 108
SIGNAL_RUN_LAST (GObject attribute), 108
T
thaw_notify() (GObject.GObject method), 107
125