mirror of
https://github.com/ToshioCP/Gtk4-tutorial.git
synced 2025-01-12 20:03:28 +01:00
377 lines
13 KiB
Markdown
377 lines
13 KiB
Markdown
Up: [README.md](../README.md), Prev: [Section 23](sec23.md), Next: [Section 25](sec25.md)
|
|
|
|
# Combine GtkDrawingArea and TfeTextView
|
|
|
|
Now, we will make a new application which has GtkDrawingArea and TfeTextView in it.
|
|
Its name is "color".
|
|
If you write a name of a color in TfeTextView and click on the `run` button, then the color of GtkDrawingArea changes to the color given by you.
|
|
|
|
![color](../image/color.png)
|
|
|
|
The following colors are available.
|
|
|
|
- white
|
|
- black
|
|
- red
|
|
- green
|
|
- blue
|
|
|
|
In addition the following two options are also available.
|
|
|
|
- light: Make the color of the drawing area lighter.
|
|
- dark: Make the color of the drawing area darker.
|
|
|
|
This application can only do very simple things.
|
|
However, it tells us that if we add powerful parser to it, we will be able to make it more efficient.
|
|
I want to show it to you in the later section by making a turtle graphics language like Logo program language.
|
|
|
|
In this section, we focus on how to bind the two objects.
|
|
|
|
## Color.ui and color.gresource.xml
|
|
|
|
First, We need to make the ui file of the widgets.
|
|
The image in the previous subsection gives us the structure of the widgets.
|
|
Title bar, four buttons in the tool bar and two widgets textview and drawing area.
|
|
The ui file is as follows.
|
|
|
|
~~~xml
|
|
1 <?xml version="1.0" encoding="UTF-8"?>
|
|
2 <interface>
|
|
3 <object class="GtkApplicationWindow" id="win">
|
|
4 <property name="title">color changer</property>
|
|
5 <property name="default-width">600</property>
|
|
6 <property name="default-height">400</property>
|
|
7 <child>
|
|
8 <object class="GtkBox" id="boxv">
|
|
9 <property name="orientation">GTK_ORIENTATION_VERTICAL</property>
|
|
10 <child>
|
|
11 <object class="GtkBox" id="boxh1">
|
|
12 <property name="orientation">GTK_ORIENTATION_HORIZONTAL</property>
|
|
13 <child>
|
|
14 <object class="GtkLabel" id="dmy1">
|
|
15 <property name="width-chars">10</property>
|
|
16 </object>
|
|
17 </child>
|
|
18 <child>
|
|
19 <object class="GtkButton" id="btnr">
|
|
20 <property name="label">Run</property>
|
|
21 <signal name="clicked" handler="run_cb"></signal>
|
|
22 </object>
|
|
23 </child>
|
|
24 <child>
|
|
25 <object class="GtkButton" id="btno">
|
|
26 <property name="label">Open</property>
|
|
27 <signal name="clicked" handler="open_cb"></signal>
|
|
28 </object>
|
|
29 </child>
|
|
30 <child>
|
|
31 <object class="GtkLabel" id="dmy2">
|
|
32 <property name="hexpand">TRUE</property>
|
|
33 </object>
|
|
34 </child>
|
|
35 <child>
|
|
36 <object class="GtkButton" id="btns">
|
|
37 <property name="label">Save</property>
|
|
38 <signal name="clicked" handler="save_cb"></signal>
|
|
39 </object>
|
|
40 </child>
|
|
41 <child>
|
|
42 <object class="GtkButton" id="btnc">
|
|
43 <property name="label">Close</property>
|
|
44 <signal name="clicked" handler="close_cb"></signal>
|
|
45 </object>
|
|
46 </child>
|
|
47 <child>
|
|
48 <object class="GtkLabel" id="dmy3">
|
|
49 <property name="width-chars">10</property>
|
|
50 </object>
|
|
51 </child>
|
|
52 </object>
|
|
53 </child>
|
|
54 <child>
|
|
55 <object class="GtkBox" id="boxh2">
|
|
56 <property name="orientation">GTK_ORIENTATION_HORIZONTAL</property>
|
|
57 <property name="homogeneous">TRUE</property>
|
|
58 <child>
|
|
59 <object class="GtkScrolledWindow" id="scr">
|
|
60 <property name="hexpand">TRUE</property>
|
|
61 <property name="vexpand">TRUE</property>
|
|
62 <child>
|
|
63 <object class="TfeTextView" id="tv">
|
|
64 <property name="wrap-mode">GTK_WRAP_WORD_CHAR</property>
|
|
65 </object>
|
|
66 </child>
|
|
67 </object>
|
|
68 </child>
|
|
69 <child>
|
|
70 <object class="GtkDrawingArea" id="da">
|
|
71 <property name="hexpand">TRUE</property>
|
|
72 <property name="vexpand">TRUE</property>
|
|
73 </object>
|
|
74 </child>
|
|
75 </object>
|
|
76 </child>
|
|
77 </object>
|
|
78 </child>
|
|
79 </object>
|
|
80 </interface>
|
|
~~~
|
|
|
|
- 10-53: This part is the tool bar which has four buttons, `Run`, `Open`, `Save` and `Close`.
|
|
This is similar to the toolbar of tfe text editor in [Section 9](sec9.md).
|
|
There are two differences.
|
|
`Run` button replaces `New` button.
|
|
A signal element is added to each button object.
|
|
It has "name" attribute which is a signal name and "handler" attribute which is the name of its signal handler function.
|
|
Options "-WI, --export-dynamic" CFLAG is necessary when you compile the application.
|
|
You can achieve this by adding "export_dynamic: true" argument to executable function in `meson.build`.
|
|
And be careful that the handler must be defined without 'static' class.
|
|
- 54-76: Puts GtkScrolledWindow and GtkDrawingArea into GtkBox.
|
|
GtkBox has "homogeneous property" with TRUE value, so the two children have the same width in the box.
|
|
TfeTextView is a child of GtkScrolledWindow.
|
|
|
|
The xml file for the resource compiler is almost same as before.
|
|
Just substitute "color" for "tfe".
|
|
|
|
~~~xml
|
|
1 <?xml version="1.0" encoding="UTF-8"?>
|
|
2 <gresources>
|
|
3 <gresource prefix="/com/github/ToshioCP/color">
|
|
4 <file>color.ui</file>
|
|
5 </gresource>
|
|
6 </gresources>
|
|
~~~
|
|
|
|
## Tfetextview.h, tfetextview.c and color.h
|
|
|
|
First two files are the same as before.
|
|
Color.h just includes tfetextview.h.
|
|
|
|
~~~C
|
|
1 #include <gtk/gtk.h>
|
|
2
|
|
3 #include "../tfetextview/tfetextview.h"
|
|
~~~
|
|
|
|
## Colorapplication.c
|
|
|
|
This is the main file.
|
|
It deals with:
|
|
|
|
- Building widgets by GtkBuilder.
|
|
- Setting a drawing function of GtkDrawingArea.
|
|
And connecting a handler to "resize" signal on GtkDrawingArea.
|
|
- Implementing each call back functions.
|
|
Particularly, `Run` signal handler is the point in this program.
|
|
|
|
The following is `colorapplication.c`.
|
|
|
|
~~~C
|
|
1 #include "color.h"
|
|
2
|
|
3 static GtkWidget *win;
|
|
4 static GtkWidget *tv;
|
|
5 static GtkWidget *da;
|
|
6
|
|
7 static cairo_surface_t *surface = NULL;
|
|
8
|
|
9 static void
|
|
10 run (void) {
|
|
11 GtkTextBuffer *tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
12 GtkTextIter start_iter;
|
|
13 GtkTextIter end_iter;
|
|
14 char *contents;
|
|
15 cairo_t *cr;
|
|
16
|
|
17 gtk_text_buffer_get_bounds (tb, &start_iter, &end_iter);
|
|
18 contents = gtk_text_buffer_get_text (tb, &start_iter, &end_iter, FALSE);
|
|
19 if (surface) {
|
|
20 cr = cairo_create (surface);
|
|
21 if (g_strcmp0 ("red", contents) == 0)
|
|
22 cairo_set_source_rgb (cr, 1, 0, 0);
|
|
23 else if (g_strcmp0 ("green", contents) == 0)
|
|
24 cairo_set_source_rgb (cr, 0, 1, 0);
|
|
25 else if (g_strcmp0 ("blue", contents) == 0)
|
|
26 cairo_set_source_rgb (cr, 0, 0, 1);
|
|
27 else if (g_strcmp0 ("white", contents) == 0)
|
|
28 cairo_set_source_rgb (cr, 1, 1, 1);
|
|
29 else if (g_strcmp0 ("black", contents) == 0)
|
|
30 cairo_set_source_rgb (cr, 0, 0, 0);
|
|
31 else if (g_strcmp0 ("light", contents) == 0)
|
|
32 cairo_set_source_rgba (cr, 1, 1, 1, 0.5);
|
|
33 else if (g_strcmp0 ("dark", contents) == 0)
|
|
34 cairo_set_source_rgba (cr, 0, 0, 0, 0.5);
|
|
35 else
|
|
36 cairo_set_source_surface (cr, surface, 0, 0);
|
|
37 cairo_paint (cr);
|
|
38 cairo_destroy (cr);
|
|
39 }
|
|
40 g_free (contents);
|
|
41 }
|
|
42
|
|
43 void
|
|
44 run_cb (GtkWidget *btnr) {
|
|
45 run ();
|
|
46 gtk_widget_queue_draw (GTK_WIDGET (da));
|
|
47 }
|
|
48
|
|
49 void
|
|
50 open_cb (GtkWidget *btno) {
|
|
51 tfe_text_view_open (TFE_TEXT_VIEW (tv), GTK_WINDOW (win));
|
|
52 }
|
|
53
|
|
54 void
|
|
55 save_cb (GtkWidget *btns) {
|
|
56 tfe_text_view_save (TFE_TEXT_VIEW (tv));
|
|
57 }
|
|
58
|
|
59 void
|
|
60 close_cb (GtkWidget *btnc) {
|
|
61 if (surface)
|
|
62 cairo_surface_destroy (surface);
|
|
63 gtk_window_destroy (GTK_WINDOW (win));
|
|
64 }
|
|
65
|
|
66 static void
|
|
67 resize_cb (GtkDrawingArea *drawing_area, int width, int height, gpointer user_data) {
|
|
68 if (surface)
|
|
69 cairo_surface_destroy (surface);
|
|
70 surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, width, height);
|
|
71 run ();
|
|
72 }
|
|
73
|
|
74 static void
|
|
75 draw_func (GtkDrawingArea *drawing_area, cairo_t *cr, int width, int height, gpointer user_data) {
|
|
76 if (surface) {
|
|
77 cairo_set_source_surface (cr, surface, 0, 0);
|
|
78 cairo_paint (cr);
|
|
79 }
|
|
80 }
|
|
81
|
|
82 static void
|
|
83 app_activate (GApplication *application) {
|
|
84 gtk_widget_show (win);
|
|
85 }
|
|
86
|
|
87 static void
|
|
88 app_startup (GApplication *application) {
|
|
89 GtkApplication *app = GTK_APPLICATION (application);
|
|
90 GtkBuilder *build;
|
|
91
|
|
92 build = gtk_builder_new_from_resource ("/com/github/ToshioCP/color/color.ui");
|
|
93 win = GTK_WIDGET (gtk_builder_get_object (build, "win"));
|
|
94 gtk_window_set_application (GTK_WINDOW (win), app);
|
|
95 tv = GTK_WIDGET (gtk_builder_get_object (build, "tv"));
|
|
96 da = GTK_WIDGET (gtk_builder_get_object (build, "da"));
|
|
97 g_object_unref(build);
|
|
98 g_signal_connect (GTK_DRAWING_AREA (da), "resize", G_CALLBACK (resize_cb), NULL);
|
|
99 gtk_drawing_area_set_draw_func (GTK_DRAWING_AREA (da), draw_func, NULL, NULL);
|
|
100
|
|
101 GdkDisplay *display;
|
|
102
|
|
103 display = gtk_widget_get_display (GTK_WIDGET (win));
|
|
104 GtkCssProvider *provider = gtk_css_provider_new ();
|
|
105 gtk_css_provider_load_from_data (provider, "textview {padding: 10px; font-family: monospace; font-size: 12pt;}", -1);
|
|
106 gtk_style_context_add_provider_for_display (display, GTK_STYLE_PROVIDER (provider), GTK_STYLE_PROVIDER_PRIORITY_USER);
|
|
107 }
|
|
108
|
|
109 #define APPLICATION_ID "com.github.ToshioCP.color"
|
|
110
|
|
111 int
|
|
112 main (int argc, char **argv) {
|
|
113 GtkApplication *app;
|
|
114 int stat;
|
|
115
|
|
116 app = gtk_application_new (APPLICATION_ID, G_APPLICATION_FLAGS_NONE);
|
|
117
|
|
118 g_signal_connect (app, "startup", G_CALLBACK (app_startup), NULL);
|
|
119 g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
|
|
120
|
|
121 stat =g_application_run (G_APPLICATION (app), argc, argv);
|
|
122 g_object_unref (app);
|
|
123 return stat;
|
|
124 }
|
|
125
|
|
~~~
|
|
|
|
- 109-124: The function `main` is almost same as before but there are some differences.
|
|
The application ID is "com.github.ToshioCP.color".
|
|
`G_APPLICATION_FLAGS_NONE` is specified so no open signal handler is necessary.
|
|
- 87-107: Startup handler.
|
|
- 92-97: Builds widgets.
|
|
The pointers of the top window, TfeTextView and GtkDrawingArea objects are stored to static variables `win`, `tv` and `da` respectively.
|
|
This is because these objects are often used in handlers.
|
|
They never be rewritten so they're thread safe.
|
|
- 98: connects "resize" signal and the handler.
|
|
- 99: sets the drawing function.
|
|
- 82-85: Activate handler, which just shows the widgets.
|
|
- 74-80: The drawing function.
|
|
It just copies `surface` to destination.
|
|
- 66-72: Resize handler.
|
|
Re-creates the surface to fit its width and height for the drawing area and paints by calling the function `run`.
|
|
- 59-64: Close handler.
|
|
It destroys `surface` if it exists.
|
|
Then it destroys the top-level window and quits the application.
|
|
- 49-57: Open and save handler.
|
|
They just call the corresponding functions of TfeTextView.
|
|
- 43-47: Run handler.
|
|
It calls run function to paint the surface.
|
|
After that `gtk_widget_queue_draw` is called.
|
|
This function adds the widget (GtkDrawingArea) to the queue to be redrawn.
|
|
It is important to know that the window is redrawn whenever it is necessary.
|
|
For example, when another window is moved and uncovers part of the widget, or when the window containing it is resized.
|
|
But repainting `surface` is not automatically notified to gtk.
|
|
Therefore, you need to call `gtk_widget_queue_draw` to redraw the widget.
|
|
- 9-41: Run function paints the surface.
|
|
First, it gets the contents of GtkTextBuffer.
|
|
Then it compares it to "red", "green" and so on.
|
|
If it matches the color, then the surface is painted the color.
|
|
If it matches "light" or "dark", then the color of the surface is lightened or darkened respectively.
|
|
Alpha channel is used.
|
|
|
|
## Meson.build
|
|
|
|
This file is almost same as before.
|
|
An argument "export_dynamic: true" is added to executable function.
|
|
|
|
~~~meson
|
|
1 project('color', 'c')
|
|
2
|
|
3 gtkdep = dependency('gtk4')
|
|
4
|
|
5 gnome=import('gnome')
|
|
6 resources = gnome.compile_resources('resources','color.gresource.xml')
|
|
7
|
|
8 sourcefiles=files('colorapplication.c', '../tfetextview/tfetextview.c')
|
|
9
|
|
10 executable('color', sourcefiles, resources, dependencies: gtkdep, export_dynamic: true)
|
|
~~~
|
|
|
|
## Compile and execute it
|
|
|
|
First you need to export some variables (refer to [Section 2](sec2.md)) if you've installed GTK 4 from the source.
|
|
If you've installed GTK 4 from the distribution packages, you don't need to do this.
|
|
|
|
$ . env.sh
|
|
|
|
Then type the following to compile it.
|
|
|
|
$ meson _build
|
|
$ ninja -C _build
|
|
|
|
The application is made in `_build` directory.
|
|
Type the following to execute it.
|
|
|
|
$ _build/color
|
|
|
|
Type "red", "green", "blue", "white", black", "light" or "dark" in the TfeTextView.
|
|
Then, click on `Run` button.
|
|
Make sure the color of GtkDrawingArea changes.
|
|
|
|
In this program TfeTextView is used to change the color.
|
|
You can use buttons or menus instead of textview.
|
|
Probably it is more appropriate.
|
|
Using textview is unnatural.
|
|
It is a good practice to make such application by yourself.
|
|
|
|
Up: [README.md](../README.md), Prev: [Section 23](sec23.md), Next: [Section 25](sec25.md)
|