2022-12-21 22:31:33 +09:00
|
|
|
Up: [README.md](../README.md), Prev: [Section 4](sec4.md), Next: [Section 6](sec6.md)
|
2021-01-11 11:15:04 +09:00
|
|
|
|
2021-01-13 13:29:35 +09:00
|
|
|
# Widgets (2)
|
2020-12-21 21:12:05 +09:00
|
|
|
|
2022-05-02 08:14:09 +09:00
|
|
|
## GtkTextView, GtkTextBuffer and GtkScrolledWindow
|
2020-12-21 21:12:05 +09:00
|
|
|
|
2021-01-13 13:29:35 +09:00
|
|
|
### GtkTextView and GtkTextBuffer
|
2020-12-21 21:12:05 +09:00
|
|
|
|
2022-05-02 08:14:09 +09:00
|
|
|
GtkTextView is a widget for multi-line text editing.
|
2021-01-13 13:29:35 +09:00
|
|
|
GtkTextBuffer is a text buffer which is connected to GtkTextView.
|
2021-06-14 17:10:23 +09:00
|
|
|
See the sample program `tfv1.c` below.
|
2020-12-22 11:30:06 +09:00
|
|
|
|
2021-01-25 18:35:49 +09:00
|
|
|
~~~C
|
|
|
|
1 #include <gtk/gtk.h>
|
|
|
|
2
|
|
|
|
3 static void
|
2022-12-21 22:31:33 +09:00
|
|
|
4 app_activate (GApplication *app) {
|
2021-01-25 18:35:49 +09:00
|
|
|
5 GtkWidget *win;
|
|
|
|
6 GtkWidget *tv;
|
|
|
|
7 GtkTextBuffer *tb;
|
|
|
|
8 gchar *text;
|
|
|
|
9
|
2022-03-04 17:09:56 +09:00
|
|
|
10 text =
|
2022-03-05 01:08:31 +10:30
|
|
|
11 "Once upon a time, there was an old man who was called Taketori-no-Okina. "
|
|
|
|
12 "It is a japanese word that means a man whose work is making bamboo baskets.\n"
|
|
|
|
13 "One day, he went into a mountain and found a shining bamboo. "
|
|
|
|
14 "\"What a mysterious bamboo it is!,\" he said. "
|
|
|
|
15 "He cut it, then there was a small cute baby girl in it. "
|
|
|
|
16 "The girl was shining faintly. "
|
|
|
|
17 "He thought this baby girl is a gift from Heaven and took her home.\n"
|
2022-12-21 22:31:33 +09:00
|
|
|
18 "His wife was surprized at his story. "
|
2022-03-05 01:08:31 +10:30
|
|
|
19 "They were very happy because they had no children. "
|
|
|
|
20 ;
|
2021-01-25 18:35:49 +09:00
|
|
|
21 win = gtk_application_window_new (GTK_APPLICATION (app));
|
|
|
|
22 gtk_window_set_title (GTK_WINDOW (win), "Taketori");
|
|
|
|
23 gtk_window_set_default_size (GTK_WINDOW (win), 400, 300);
|
|
|
|
24
|
|
|
|
25 tv = gtk_text_view_new ();
|
|
|
|
26 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
|
|
27 gtk_text_buffer_set_text (tb, text, -1);
|
|
|
|
28 gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (tv), GTK_WRAP_WORD_CHAR);
|
|
|
|
29
|
|
|
|
30 gtk_window_set_child (GTK_WINDOW (win), tv);
|
|
|
|
31
|
2023-01-03 15:30:06 +09:00
|
|
|
32 gtk_window_present (GTK_WINDOW (win));
|
2021-01-25 18:35:49 +09:00
|
|
|
33 }
|
|
|
|
34
|
|
|
|
35 int
|
|
|
|
36 main (int argc, char **argv) {
|
|
|
|
37 GtkApplication *app;
|
|
|
|
38 int stat;
|
|
|
|
39
|
2022-12-21 22:31:33 +09:00
|
|
|
40 app = gtk_application_new ("com.github.ToshioCP.tfv1", G_APPLICATION_DEFAULT_FLAGS);
|
2021-06-14 17:10:23 +09:00
|
|
|
41 g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
|
2022-03-05 01:08:31 +10:30
|
|
|
42 stat = g_application_run (G_APPLICATION (app), argc, argv);
|
2021-01-25 18:35:49 +09:00
|
|
|
43 g_object_unref (app);
|
|
|
|
44 return stat;
|
|
|
|
45 }
|
|
|
|
~~~
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
Look at line 25.
|
2021-06-14 17:10:23 +09:00
|
|
|
A GtkTextView instance is created and its pointer is assigned to `tv`.
|
|
|
|
When the GtkTextView instance is created, a GtkTextBuffer instance is also created and connected to the GtkTextView automatically.
|
|
|
|
"GtkTextBuffer instance" will be referred to simply as "GtkTextBuffer" or "buffer".
|
2022-12-21 22:31:33 +09:00
|
|
|
In the next line, the pointer to the buffer is assigned to `tb`.
|
2021-01-13 13:29:35 +09:00
|
|
|
Then, the text from line 10 to 20 is assigned to the buffer.
|
2022-12-21 22:31:33 +09:00
|
|
|
If the third argument of `gtk_text_buffer_set_text` is a positive integer, it is the length of the text.
|
|
|
|
It it is -1, the string terminates with NULL.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
GtkTextView has a wrap mode.
|
2021-02-06 17:26:57 +09:00
|
|
|
When it is set to `GTK_WRAP_WORD_CHAR`, text wraps in between words, or if that is not enough, also between graphemes.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
2022-12-21 22:31:33 +09:00
|
|
|
Wrap mode is written in [Gtk\_WrapMode](https://docs.gtk.org/gtk4/enum.WrapMode.html) in the GTK 4 API document.
|
|
|
|
|
2021-02-06 17:26:57 +09:00
|
|
|
In line 30, `tv` is added to `win` as a child.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
Now compile and run it.
|
|
|
|
|
2022-12-21 22:31:33 +09:00
|
|
|
```
|
|
|
|
$ cd src/tfv
|
|
|
|
$ comp tfv1
|
|
|
|
$ ./a.out
|
|
|
|
```
|
|
|
|
|
2021-01-13 20:59:15 +09:00
|
|
|
![GtkTextView](../image/screenshot_tfv1.png)
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
There's an I-beam pointer in the window.
|
2022-05-02 08:14:09 +09:00
|
|
|
You can add or delete any characters on the GtkTextView,
|
2022-03-04 17:09:56 +09:00
|
|
|
and your changes are kept in the GtkTextBuffer.
|
|
|
|
If you add more characters beyond the limit of the window, the height increases and the window extends.
|
|
|
|
If the height gets bigger than the height of the display screen, you won't be
|
2022-12-21 22:31:33 +09:00
|
|
|
able to control the size of the window or change it back to the original size.
|
|
|
|
This is a problem, that is to say a bug.
|
|
|
|
This can be solved by adding a GtkScrolledWindow between the GtkApplicationWindow and GtkTextView.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
### GtkScrolledWindow
|
|
|
|
|
|
|
|
What we need to do is:
|
|
|
|
|
2022-12-21 22:31:33 +09:00
|
|
|
- Create a GtkScrolledWindow and insert it as a child of the GtkApplicationWindow
|
2022-03-04 17:09:56 +09:00
|
|
|
- Insert the GtkTextView widget to the GtkScrolledWindow as a child.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
|
|
|
Modify `tfv1.c` and save it as `tfv2.c`.
|
2022-12-21 22:31:33 +09:00
|
|
|
There is only a few difference between these two files.
|
2021-01-13 13:29:35 +09:00
|
|
|
|
2021-02-08 22:24:54 +09:00
|
|
|
~~~
|
|
|
|
$ cd tfv; diff tfv1.c tfv2.c
|
|
|
|
5a6
|
|
|
|
> GtkWidget *scr;
|
|
|
|
24a26,28
|
|
|
|
> scr = gtk_scrolled_window_new ();
|
|
|
|
> gtk_window_set_child (GTK_WINDOW (win), scr);
|
|
|
|
>
|
|
|
|
30c34
|
|
|
|
< gtk_window_set_child (GTK_WINDOW (win), tv);
|
|
|
|
---
|
|
|
|
> gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scr), tv);
|
|
|
|
40c44
|
2022-12-21 22:31:33 +09:00
|
|
|
< app = gtk_application_new ("com.github.ToshioCP.tfv1", G_APPLICATION_DEFAULT_FLAGS);
|
2021-02-08 22:24:54 +09:00
|
|
|
---
|
2022-12-21 22:31:33 +09:00
|
|
|
> app = gtk_application_new ("com.github.ToshioCP.tfv2", G_APPLICATION_DEFAULT_FLAGS);
|
2021-02-08 22:24:54 +09:00
|
|
|
~~~
|
2021-01-13 13:29:35 +09:00
|
|
|
|
2022-12-21 22:31:33 +09:00
|
|
|
The whole code of `tfv2.c` is as follows.
|
2020-12-22 11:30:06 +09:00
|
|
|
|
2021-01-25 18:35:49 +09:00
|
|
|
~~~C
|
|
|
|
1 #include <gtk/gtk.h>
|
|
|
|
2
|
|
|
|
3 static void
|
2022-12-21 22:31:33 +09:00
|
|
|
4 app_activate (GApplication *app) {
|
2021-01-25 18:35:49 +09:00
|
|
|
5 GtkWidget *win;
|
|
|
|
6 GtkWidget *scr;
|
|
|
|
7 GtkWidget *tv;
|
|
|
|
8 GtkTextBuffer *tb;
|
|
|
|
9 gchar *text;
|
|
|
|
10
|
2022-03-04 17:09:56 +09:00
|
|
|
11 text =
|
2022-03-05 01:08:31 +10:30
|
|
|
12 "Once upon a time, there was an old man who was called Taketori-no-Okina. "
|
|
|
|
13 "It is a japanese word that means a man whose work is making bamboo baskets.\n"
|
|
|
|
14 "One day, he went into a mountain and found a shining bamboo. "
|
|
|
|
15 "\"What a mysterious bamboo it is!,\" he said. "
|
|
|
|
16 "He cut it, then there was a small cute baby girl in it. "
|
|
|
|
17 "The girl was shining faintly. "
|
|
|
|
18 "He thought this baby girl is a gift from Heaven and took her home.\n"
|
2022-12-21 22:31:33 +09:00
|
|
|
19 "His wife was surprized at his story. "
|
2022-03-05 01:08:31 +10:30
|
|
|
20 "They were very happy because they had no children. "
|
|
|
|
21 ;
|
2021-01-25 18:35:49 +09:00
|
|
|
22 win = gtk_application_window_new (GTK_APPLICATION (app));
|
|
|
|
23 gtk_window_set_title (GTK_WINDOW (win), "Taketori");
|
|
|
|
24 gtk_window_set_default_size (GTK_WINDOW (win), 400, 300);
|
|
|
|
25
|
|
|
|
26 scr = gtk_scrolled_window_new ();
|
|
|
|
27 gtk_window_set_child (GTK_WINDOW (win), scr);
|
|
|
|
28
|
|
|
|
29 tv = gtk_text_view_new ();
|
|
|
|
30 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv));
|
|
|
|
31 gtk_text_buffer_set_text (tb, text, -1);
|
|
|
|
32 gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (tv), GTK_WRAP_WORD_CHAR);
|
|
|
|
33
|
|
|
|
34 gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scr), tv);
|
|
|
|
35
|
2023-01-03 15:30:06 +09:00
|
|
|
36 gtk_window_present (GTK_WINDOW (win));
|
2021-01-25 18:35:49 +09:00
|
|
|
37 }
|
|
|
|
38
|
|
|
|
39 int
|
|
|
|
40 main (int argc, char **argv) {
|
|
|
|
41 GtkApplication *app;
|
|
|
|
42 int stat;
|
|
|
|
43
|
2022-12-21 22:31:33 +09:00
|
|
|
44 app = gtk_application_new ("com.github.ToshioCP.tfv2", G_APPLICATION_DEFAULT_FLAGS);
|
2021-06-14 17:10:23 +09:00
|
|
|
45 g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
|
2022-03-05 01:08:31 +10:30
|
|
|
46 stat = g_application_run (G_APPLICATION (app), argc, argv);
|
2021-01-25 18:35:49 +09:00
|
|
|
47 g_object_unref (app);
|
|
|
|
48 return stat;
|
|
|
|
49 }
|
|
|
|
~~~
|
2021-01-13 13:29:35 +09:00
|
|
|
|
2022-03-04 17:09:56 +09:00
|
|
|
Compile and run it.
|
2021-01-11 11:15:04 +09:00
|
|
|
|
2022-12-21 22:31:33 +09:00
|
|
|
Now, the window doesn't extend even if you type a lot of characters,
|
|
|
|
it just scrolls.
|
|
|
|
|
|
|
|
Up: [README.md](../README.md), Prev: [Section 4](sec4.md), Next: [Section 6](sec6.md)
|